2014-08-05 00:50:25 -04:00
|
|
|
package graph
|
|
|
|
|
|
|
|
import (
|
2015-06-05 18:31:10 -04:00
|
|
|
"fmt"
|
2014-08-05 00:50:25 -04:00
|
|
|
"strings"
|
|
|
|
|
2015-04-03 11:31:30 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
2015-07-20 13:57:15 -04:00
|
|
|
"github.com/docker/docker/image"
|
2015-02-26 21:23:50 -05:00
|
|
|
"github.com/docker/docker/utils"
|
2014-08-05 00:50:25 -04:00
|
|
|
)
|
|
|
|
|
2015-06-05 18:31:10 -04:00
|
|
|
// WalkHistory calls the handler function for each image in the
|
|
|
|
// provided images lineage starting from immediate parent.
|
2015-07-20 13:57:15 -04:00
|
|
|
func (graph *Graph) WalkHistory(img *image.Image, handler func(image.Image) error) (err error) {
|
2015-06-05 18:31:10 -04:00
|
|
|
currentImg := img
|
|
|
|
for currentImg != nil {
|
|
|
|
if handler != nil {
|
2015-06-05 18:32:31 -04:00
|
|
|
if err := handler(*currentImg); err != nil {
|
2015-06-05 18:31:10 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
currentImg, err = graph.GetParent(currentImg)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Error while getting parent image: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-09-06 08:46:47 -04:00
|
|
|
// depth returns the number of parents for the current image
|
2015-07-20 13:57:15 -04:00
|
|
|
func (graph *Graph) depth(img *image.Image) (int, error) {
|
2015-06-05 18:31:10 -04:00
|
|
|
var (
|
|
|
|
count = 0
|
|
|
|
parent = img
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
for parent != nil {
|
|
|
|
count++
|
2015-09-06 08:46:47 -04:00
|
|
|
if parent, err = graph.GetParent(parent); err != nil {
|
2015-06-05 18:31:10 -04:00
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return count, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set the max depth to the aufs default that most
|
|
|
|
// kernels are compiled with
|
|
|
|
// For more information see: http://sourceforge.net/p/aufs/aufs3-standalone/ci/aufs3.12/tree/config.mk
|
|
|
|
const MaxImageDepth = 127
|
|
|
|
|
|
|
|
// CheckDepth returns an error if the depth of an image, as returned
|
|
|
|
// by ImageDepth, is too large to support creating a container from it
|
|
|
|
// on this daemon.
|
2015-07-20 13:57:15 -04:00
|
|
|
func (graph *Graph) CheckDepth(img *image.Image) error {
|
2015-06-05 18:31:10 -04:00
|
|
|
// We add 2 layers to the depth because the container's rw and
|
|
|
|
// init layer add to the restriction
|
|
|
|
depth, err := graph.depth(img)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if depth+2 >= MaxImageDepth {
|
|
|
|
return fmt.Errorf("Cannot create container with more than %d parents", MaxImageDepth)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-07-29 19:45:47 -04:00
|
|
|
// History returns a slice of ImageHistory structures for the specified image
|
|
|
|
// name by walking the image lineage.
|
2015-04-10 06:55:07 -04:00
|
|
|
func (s *TagStore) History(name string) ([]*types.ImageHistory, error) {
|
2014-08-05 00:50:25 -04:00
|
|
|
foundImage, err := s.LookupImage(name)
|
|
|
|
if err != nil {
|
2015-04-10 06:55:07 -04:00
|
|
|
return nil, err
|
2014-08-05 00:50:25 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
lookupMap := make(map[string][]string)
|
|
|
|
for name, repository := range s.Repositories {
|
|
|
|
for tag, id := range repository {
|
|
|
|
// If the ID already has a reverse lookup, do not update it unless for "latest"
|
|
|
|
if _, exists := lookupMap[id]; !exists {
|
|
|
|
lookupMap[id] = []string{}
|
|
|
|
}
|
2015-02-26 21:23:50 -05:00
|
|
|
lookupMap[id] = append(lookupMap[id], utils.ImageReference(name, tag))
|
2014-08-05 00:50:25 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-10 06:55:07 -04:00
|
|
|
history := []*types.ImageHistory{}
|
2015-04-03 11:31:30 -04:00
|
|
|
|
2015-07-20 13:57:15 -04:00
|
|
|
err = s.graph.WalkHistory(foundImage, func(img image.Image) error {
|
2015-04-10 06:55:07 -04:00
|
|
|
history = append(history, &types.ImageHistory{
|
2015-04-03 11:31:30 -04:00
|
|
|
ID: img.ID,
|
2015-05-14 20:31:34 -04:00
|
|
|
Created: img.Created.Unix(),
|
2015-04-10 20:05:21 -04:00
|
|
|
CreatedBy: strings.Join(img.ContainerConfig.Cmd.Slice(), " "),
|
2015-04-03 11:31:30 -04:00
|
|
|
Tags: lookupMap[img.ID],
|
|
|
|
Size: img.Size,
|
2015-01-04 01:47:01 -05:00
|
|
|
Comment: img.Comment,
|
2015-04-03 11:31:30 -04:00
|
|
|
})
|
2014-08-05 00:50:25 -04:00
|
|
|
return nil
|
|
|
|
})
|
2015-04-03 11:31:30 -04:00
|
|
|
|
2015-04-10 06:55:07 -04:00
|
|
|
return history, err
|
2014-08-05 00:50:25 -04:00
|
|
|
}
|
2015-06-05 18:31:10 -04:00
|
|
|
|
2015-07-29 19:45:47 -04:00
|
|
|
// GetParent returns the parent image for the specified image.
|
2015-07-20 13:57:15 -04:00
|
|
|
func (graph *Graph) GetParent(img *image.Image) (*image.Image, error) {
|
2015-06-05 18:31:10 -04:00
|
|
|
if img.Parent == "" {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
return graph.Get(img.Parent)
|
|
|
|
}
|
|
|
|
|
2015-07-29 19:45:47 -04:00
|
|
|
// GetParentsSize returns the combined size of all parent images. If there is
|
|
|
|
// no parent image or it's unavailable, it returns 0.
|
|
|
|
func (graph *Graph) GetParentsSize(img *image.Image) int64 {
|
2015-06-05 18:31:10 -04:00
|
|
|
parentImage, err := graph.GetParent(img)
|
|
|
|
if err != nil || parentImage == nil {
|
2015-07-29 19:45:47 -04:00
|
|
|
return 0
|
2015-06-05 18:31:10 -04:00
|
|
|
}
|
2015-07-29 19:45:47 -04:00
|
|
|
return parentImage.Size + graph.GetParentsSize(parentImage)
|
2015-06-05 18:31:10 -04:00
|
|
|
}
|