123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102 |
- /*
- 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 dockertools
- import (
- "fmt"
- "sync"
- "github.com/golang/glog"
- dockertypes "github.com/docker/engine-api/types"
- runtime "k8s.io/kubernetes/pkg/kubelet/container"
- )
- // imageStatsProvider exposes stats about all images currently available.
- type imageStatsProvider struct {
- sync.Mutex
- // layers caches the current layers, key is the layer ID.
- layers map[string]*dockertypes.ImageHistory
- // imageToLayerIDs maps image to its layer IDs.
- imageToLayerIDs map[string][]string
- // Docker remote API client
- c DockerInterface
- }
- func newImageStatsProvider(c DockerInterface) *imageStatsProvider {
- return &imageStatsProvider{
- layers: make(map[string]*dockertypes.ImageHistory),
- imageToLayerIDs: make(map[string][]string),
- c: c,
- }
- }
- func (isp *imageStatsProvider) ImageStats() (*runtime.ImageStats, error) {
- images, err := isp.c.ListImages(dockertypes.ImageListOptions{})
- if err != nil {
- return nil, fmt.Errorf("failed to list docker images - %v", err)
- }
- // Take the lock to protect the cache
- isp.Lock()
- defer isp.Unlock()
- // Create new cache each time, this is a little more memory consuming, but:
- // * ImageStats is only called every 10 seconds
- // * We use pointers and reference to copy cache elements.
- // The memory usage should be acceptable.
- // TODO(random-liu): Add more logic to implement in place cache update.
- newLayers := make(map[string]*dockertypes.ImageHistory)
- newImageToLayerIDs := make(map[string][]string)
- for _, image := range images {
- layerIDs, ok := isp.imageToLayerIDs[image.ID]
- if !ok {
- // Get information about the various layers of the given docker image.
- history, err := isp.c.ImageHistory(image.ID)
- if err != nil {
- // Skip the image and inspect again in next ImageStats if the image is still there
- glog.V(2).Infof("failed to get history of docker image %+v - %v", image, err)
- continue
- }
- // Cache each layer
- for i := range history {
- layer := &history[i]
- key := layer.ID
- // Some of the layers are empty.
- // We are hoping that these layers are unique to each image.
- // Still keying with the CreatedBy field to be safe.
- if key == "" || key == "<missing>" {
- key = key + layer.CreatedBy
- }
- layerIDs = append(layerIDs, key)
- newLayers[key] = layer
- }
- } else {
- for _, layerID := range layerIDs {
- newLayers[layerID] = isp.layers[layerID]
- }
- }
- newImageToLayerIDs[image.ID] = layerIDs
- }
- ret := &runtime.ImageStats{}
- // Calculate the total storage bytes
- for _, layer := range newLayers {
- ret.TotalStorageBytes += uint64(layer.Size)
- }
- // Update current cache
- isp.layers = newLayers
- isp.imageToLayerIDs = newImageToLayerIDs
- return ret, nil
- }
|