2014-03-07 21:04:38 -05:00
package graph
2013-03-18 03:15:35 -04:00
import (
2015-03-30 14:19:12 -04:00
"compress/gzip"
"crypto/sha256"
2015-06-05 18:31:10 -04:00
"encoding/json"
2015-06-05 21:07:41 -04:00
"errors"
2013-03-18 03:15:35 -04:00
"fmt"
2013-05-06 14:06:44 -04:00
"io"
2013-03-18 03:15:35 -04:00
"io/ioutil"
"os"
"path/filepath"
2013-12-20 11:20:08 -05:00
"runtime"
2015-06-05 18:31:10 -04:00
"strconv"
2013-03-26 08:28:17 -04:00
"strings"
2015-04-07 22:29:29 -04:00
"sync"
2013-03-18 03:15:35 -04:00
"time"
2014-05-19 18:04:51 -04:00
2015-03-26 18:22:04 -04:00
"github.com/Sirupsen/logrus"
2015-03-30 14:19:12 -04:00
"github.com/docker/distribution/digest"
2015-02-04 16:22:38 -05:00
"github.com/docker/docker/autogen/dockerversion"
2014-07-24 18:19:50 -04:00
"github.com/docker/docker/daemon/graphdriver"
"github.com/docker/docker/image"
2014-09-30 02:23:36 -04:00
"github.com/docker/docker/pkg/archive"
2015-02-24 03:51:46 -05:00
"github.com/docker/docker/pkg/progressreader"
2015-03-17 22:18:41 -04:00
"github.com/docker/docker/pkg/streamformatter"
2015-03-24 07:25:26 -04:00
"github.com/docker/docker/pkg/stringid"
2015-05-04 18:05:54 -04:00
"github.com/docker/docker/pkg/system"
2014-07-24 18:19:50 -04:00
"github.com/docker/docker/pkg/truncindex"
"github.com/docker/docker/runconfig"
2015-07-20 21:32:14 -04:00
"github.com/vbatts/tar-split/tar/asm"
"github.com/vbatts/tar-split/tar/storage"
2013-03-18 03:15:35 -04:00
)
2015-04-07 22:29:29 -04:00
// The type is used to protect pulling or building related image
// layers from deleteing when filtered by dangling=true
// The key of layers is the images ID which is pulling or building
// The value of layers is a slice which hold layer IDs referenced to
// pulling or building images
type retainedLayers struct {
layerHolders map [ string ] map [ string ] struct { } // map[layerID]map[sessionID]
sync . Mutex
}
func ( r * retainedLayers ) Add ( sessionID string , layerIDs [ ] string ) {
r . Lock ( )
defer r . Unlock ( )
for _ , layerID := range layerIDs {
if r . layerHolders [ layerID ] == nil {
r . layerHolders [ layerID ] = map [ string ] struct { } { }
}
r . layerHolders [ layerID ] [ sessionID ] = struct { } { }
}
}
func ( r * retainedLayers ) Delete ( sessionID string , layerIDs [ ] string ) {
r . Lock ( )
defer r . Unlock ( )
for _ , layerID := range layerIDs {
holders , ok := r . layerHolders [ layerID ]
if ! ok {
continue
}
delete ( holders , sessionID )
if len ( holders ) == 0 {
delete ( r . layerHolders , layerID ) // Delete any empty reference set.
}
}
}
func ( r * retainedLayers ) Exists ( layerID string ) bool {
r . Lock ( )
_ , exists := r . layerHolders [ layerID ]
r . Unlock ( )
return exists
}
2013-03-30 03:22:24 -04:00
// A Graph is a store for versioned filesystem images and the relationship between them.
2013-03-18 03:15:35 -04:00
type Graph struct {
2015-06-04 15:58:58 -04:00
root string
idIndex * truncindex . TruncIndex
driver graphdriver . Driver
imageMutex imageMutex // protect images in driver.
2015-04-07 22:29:29 -04:00
retained * retainedLayers
2013-03-18 03:15:35 -04:00
}
2015-06-23 19:44:07 -04:00
// file names for ./graph/<ID>/
const (
jsonFileName = "json"
layersizeFileName = "layersize"
digestFileName = "checksum"
2015-06-23 19:44:49 -04:00
tarDataFileName = "tar-data.json.gz"
2015-06-23 19:44:07 -04:00
)
2015-06-05 21:07:41 -04:00
var (
// ErrDigestNotSet is used when request the digest for a layer
// but the layer has no digest value or content to compute the
// the digest.
ErrDigestNotSet = errors . New ( "digest is not set for layer" )
)
2013-03-30 03:22:24 -04:00
// NewGraph instantiates a new graph at the given root path in the filesystem.
2013-03-30 00:13:59 -04:00
// `root` will be created if it doesn't exist.
2013-11-07 15:34:01 -05:00
func NewGraph ( root string , driver graphdriver . Driver ) ( * Graph , error ) {
2013-03-18 03:15:35 -04:00
abspath , err := filepath . Abs ( root )
if err != nil {
return nil , err
}
// Create the root directory if it doesn't exists
Simplify and fix os.MkdirAll() usage
TL;DR: check for IsExist(err) after a failed MkdirAll() is both
redundant and wrong -- so two reasons to remove it.
Quoting MkdirAll documentation:
> MkdirAll creates a directory named path, along with any necessary
> parents, and returns nil, or else returns an error. If path
> is already a directory, MkdirAll does nothing and returns nil.
This means two things:
1. If a directory to be created already exists, no error is returned.
2. If the error returned is IsExist (EEXIST), it means there exists
a non-directory with the same name as MkdirAll need to use for
directory. Example: we want to MkdirAll("a/b"), but file "a"
(or "a/b") already exists, so MkdirAll fails.
The above is a theory, based on quoted documentation and my UNIX
knowledge.
3. In practice, though, current MkdirAll implementation [1] returns
ENOTDIR in most of cases described in #2, with the exception when
there is a race between MkdirAll and someone else creating the
last component of MkdirAll argument as a file. In this very case
MkdirAll() will indeed return EEXIST.
Because of #1, IsExist check after MkdirAll is not needed.
Because of #2 and #3, ignoring IsExist error is just plain wrong,
as directory we require is not created. It's cleaner to report
the error now.
Note this error is all over the tree, I guess due to copy-paste,
or trying to follow the same usage pattern as for Mkdir(),
or some not quite correct examples on the Internet.
[v2: a separate aufs commit is merged into this one]
[1] https://github.com/golang/go/blob/f9ed2f75/src/os/path.go
Signed-off-by: Kir Kolyshkin <kir@openvz.org>
2015-07-29 19:49:05 -04:00
if err := system . MkdirAll ( root , 0700 ) ; err != nil {
2013-03-18 03:15:35 -04:00
return nil , err
}
2013-11-04 18:22:34 -05:00
2013-04-01 01:11:55 -04:00
graph := & Graph {
2015-04-07 22:29:29 -04:00
root : abspath ,
idIndex : truncindex . NewTruncIndex ( [ ] string { } ) ,
driver : driver ,
retained : & retainedLayers { layerHolders : make ( map [ string ] map [ string ] struct { } ) } ,
2013-04-01 01:11:55 -04:00
}
if err := graph . restore ( ) ; err != nil {
return nil , err
}
return graph , nil
}
2015-04-07 22:29:29 -04:00
// IsHeld returns whether the given layerID is being used by an ongoing pull or build.
func ( graph * Graph ) IsHeld ( layerID string ) bool {
return graph . retained . Exists ( layerID )
}
2013-04-01 01:11:55 -04:00
func ( graph * Graph ) restore ( ) error {
2015-06-05 18:31:10 -04:00
dir , err := ioutil . ReadDir ( graph . root )
2013-04-01 01:11:55 -04:00
if err != nil {
return err
}
2014-04-11 16:39:58 -04:00
var ids = [ ] string { }
2013-04-01 01:11:55 -04:00
for _ , v := range dir {
id := v . Name ( )
2013-11-19 05:32:08 -05:00
if graph . driver . Exists ( id ) {
2014-04-11 16:39:58 -04:00
ids = append ( ids , id )
2013-11-19 05:32:08 -05:00
}
2013-04-01 01:11:55 -04:00
}
2015-06-11 14:29:29 -04:00
baseIds , err := graph . restoreBaseImages ( )
if err != nil {
return err
}
ids = append ( ids , baseIds ... )
2014-06-24 13:19:15 -04:00
graph . idIndex = truncindex . NewTruncIndex ( ids )
2015-06-10 18:18:51 -04:00
logrus . Debugf ( "Restored %d elements" , len ( ids ) )
2013-04-01 01:11:55 -04:00
return nil
2013-03-18 03:15:35 -04:00
}
2015-07-29 19:45:47 -04:00
// IsNotExist detects whether an image exists by parsing the incoming error
// message.
2015-03-27 21:07:20 -04:00
func ( graph * Graph ) IsNotExist ( err error , id string ) bool {
2015-07-29 19:45:47 -04:00
// FIXME: Implement error subclass instead of looking at the error text
// Note: This is the way golang implements os.IsNotExists on Plan9
2015-03-27 21:07:20 -04:00
return err != nil && ( strings . Contains ( strings . ToLower ( err . Error ( ) ) , "does not exist" ) || strings . Contains ( strings . ToLower ( err . Error ( ) ) , "no such" ) ) && strings . Contains ( err . Error ( ) , id )
2013-03-26 08:28:17 -04:00
}
2013-03-30 00:13:59 -04:00
// Exists returns true if an image is registered at the given id.
// If the image doesn't exist or if an error is encountered, false is returned.
2013-03-18 03:15:35 -04:00
func ( graph * Graph ) Exists ( id string ) bool {
if _ , err := graph . Get ( id ) ; err != nil {
return false
}
return true
}
2013-03-30 00:13:59 -04:00
// Get returns the image with the given id, or an error if the image doesn't exist.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) Get ( name string ) ( * image . Image , error ) {
2013-04-01 01:11:55 -04:00
id , err := graph . idIndex . Get ( name )
if err != nil {
2014-12-11 12:57:23 -05:00
return nil , fmt . Errorf ( "could not find image: %v" , err )
2013-04-01 01:11:55 -04:00
}
2015-06-05 18:31:10 -04:00
img , err := graph . loadImage ( id )
2013-03-18 03:15:35 -04:00
if err != nil {
return nil , err
}
2013-06-04 14:00:22 -04:00
if img . ID != id {
return nil , fmt . Errorf ( "Image stored at '%s' has wrong id '%s'" , id , img . ID )
2013-03-18 03:15:35 -04:00
}
2013-12-05 17:03:23 -05:00
if img . Size < 0 {
2014-09-10 23:30:52 -04:00
size , err := graph . driver . DiffSize ( img . ID , img . Parent )
2014-01-07 14:34:19 -05:00
if err != nil {
2014-09-10 23:30:52 -04:00
return nil , fmt . Errorf ( "unable to calculate size of image id %q: %s" , img . ID , err )
2013-05-13 09:10:26 -04:00
}
2013-12-05 17:03:23 -05:00
2013-11-07 15:34:01 -05:00
img . Size = size
2015-07-23 17:19:58 -04:00
if err := graph . saveSize ( graph . imageRoot ( id ) , img . Size ) ; err != nil {
2013-05-13 09:10:26 -04:00
return nil , err
}
}
2013-03-18 03:15:35 -04:00
return img , nil
}
2013-03-30 00:13:59 -04:00
// Create creates a new image and registers it in the graph.
2015-08-03 21:52:54 -04:00
func ( graph * Graph ) Create ( layerData archive . Reader , containerID , containerImage , comment , author string , containerConfig , config * runconfig . Config ) ( * image . Image , error ) {
2015-07-20 13:57:15 -04:00
img := & image . Image {
2015-03-24 07:25:26 -04:00
ID : stringid . GenerateRandomID ( ) ,
2013-04-04 21:38:43 -04:00
Comment : comment ,
2013-11-21 19:41:41 -05:00
Created : time . Now ( ) . UTC ( ) ,
2014-02-11 19:26:54 -05:00
DockerVersion : dockerversion . VERSION ,
2013-04-17 22:58:17 -04:00
Author : author ,
2013-04-25 19:48:31 -04:00
Config : config ,
2013-12-20 11:20:08 -05:00
Architecture : runtime . GOARCH ,
OS : runtime . GOOS ,
2013-03-18 03:15:35 -04:00
}
2014-05-19 18:04:51 -04:00
2014-03-07 21:04:38 -05:00
if containerID != "" {
img . Parent = containerImage
img . Container = containerID
img . ContainerConfig = * containerConfig
2013-03-22 00:13:27 -04:00
}
2014-05-19 18:04:51 -04:00
2014-10-27 14:00:29 -04:00
if err := graph . Register ( img , layerData ) ; err != nil {
2013-03-18 03:15:35 -04:00
return nil , err
}
return img , nil
}
2013-03-30 00:13:59 -04:00
// Register imports a pre-existing image into the graph.
2015-08-03 21:52:54 -04:00
func ( graph * Graph ) Register ( img * image . Image , layerData archive . Reader ) ( err error ) {
2015-07-17 10:01:52 -04:00
2015-06-04 15:58:58 -04:00
if err := image . ValidateID ( img . ID ) ; err != nil {
return err
}
// We need this entire operation to be atomic within the engine. Note that
// this doesn't mean Register is fully safe yet.
graph . imageMutex . Lock ( img . ID )
defer graph . imageMutex . Unlock ( img . ID )
2015-07-17 10:01:52 -04:00
// The returned `error` must be named in this function's signature so that
// `err` is not shadowed in this deferred cleanup.
2013-11-25 23:48:34 -05:00
defer func ( ) {
// If any error occurs, remove the new dir from the driver.
// Don't check for errors since the dir might not have been created.
if err != nil {
graph . driver . Remove ( img . ID )
}
} ( )
2015-06-04 15:58:58 -04:00
2013-03-18 03:15:35 -04:00
// (This is a convenience to save time. Race conditions are taken care of by os.Rename)
2013-06-04 14:00:22 -04:00
if graph . Exists ( img . ID ) {
return fmt . Errorf ( "Image %s already exists" , img . ID )
2013-03-18 03:15:35 -04:00
}
2013-11-19 05:32:08 -05:00
// Ensure that the image root does not exist on the filesystem
// when it is not registered in the graph.
// This is common when you switch from one graph driver to another
2015-06-05 18:31:10 -04:00
if err := os . RemoveAll ( graph . imageRoot ( img . ID ) ) ; err != nil && ! os . IsNotExist ( err ) {
2013-11-19 05:32:08 -05:00
return err
}
2013-11-25 23:48:34 -05:00
// If the driver has this ID but the graph doesn't, remove it from the driver to start fresh.
// (the graph is the source of truth).
// Ignore errors, since we don't know if the driver correctly returns ErrNotExist.
// (FIXME: make that mandatory for drivers).
graph . driver . Remove ( img . ID )
2015-06-05 18:31:10 -04:00
tmp , err := graph . mktemp ( "" )
2013-03-18 03:15:35 -04:00
defer os . RemoveAll ( tmp )
if err != nil {
2015-06-05 18:31:10 -04:00
return fmt . Errorf ( "mktemp failed: %s" , err )
2013-03-18 03:15:35 -04:00
}
2013-11-07 15:34:01 -05:00
// Create root filesystem in the driver
2015-06-11 14:29:29 -04:00
if err := createRootFilesystemInDriver ( graph , img , layerData ) ; err != nil {
return err
2013-11-07 15:34:01 -05:00
}
2015-06-11 14:29:29 -04:00
2014-09-10 23:30:52 -04:00
// Apply the diff/layer
2015-06-05 18:31:10 -04:00
if err := graph . storeImage ( img , layerData , tmp ) ; err != nil {
2013-03-18 03:15:35 -04:00
return err
}
// Commit
2015-06-05 18:31:10 -04:00
if err := os . Rename ( tmp , graph . imageRoot ( img . ID ) ) ; err != nil {
2013-03-18 03:15:35 -04:00
return err
}
2013-06-04 14:00:22 -04:00
graph . idIndex . Add ( img . ID )
2013-03-18 03:15:35 -04:00
return nil
}
2013-04-21 17:23:55 -04:00
// TempLayerArchive creates a temporary archive of the given image's filesystem layer.
// The archive is stored on disk and will be automatically deleted as soon as has been read.
2013-04-21 18:29:26 -04:00
// If output is not nil, a human-readable progress bar will be written to it.
2015-03-17 22:18:41 -04:00
func ( graph * Graph ) TempLayerArchive ( id string , sf * streamformatter . StreamFormatter , output io . Writer ) ( * archive . TempArchive , error ) {
2013-04-21 17:23:55 -04:00
image , err := graph . Get ( id )
if err != nil {
return nil , err
}
2015-06-05 18:31:10 -04:00
tmp , err := graph . mktemp ( "" )
2013-04-21 17:23:55 -04:00
if err != nil {
return nil , err
}
2015-06-05 18:31:10 -04:00
a , err := graph . TarLayer ( image )
2013-04-21 17:23:55 -04:00
if err != nil {
return nil , err
}
2015-02-24 03:51:46 -05:00
progressReader := progressreader . New ( progressreader . Config {
In : a ,
Out : output ,
Formatter : sf ,
Size : 0 ,
NewLines : false ,
2015-03-24 07:25:26 -04:00
ID : stringid . TruncateID ( id ) ,
2015-02-24 03:51:46 -05:00
Action : "Buffering to disk" ,
} )
defer progressReader . Close ( )
return archive . NewTempArchive ( progressReader , tmp )
2013-04-21 17:23:55 -04:00
}
2015-06-05 18:31:10 -04:00
// mktemp creates a temporary sub-directory inside the graph's filesystem.
func ( graph * Graph ) mktemp ( id string ) ( string , error ) {
2015-07-28 20:19:17 -04:00
dir := filepath . Join ( graph . root , "_tmp" , stringid . GenerateNonCryptoID ( ) )
2015-05-04 18:05:54 -04:00
if err := system . MkdirAll ( dir , 0700 ) ; err != nil {
2013-11-08 19:53:58 -05:00
return "" , err
2013-03-18 03:15:35 -04:00
}
2013-11-07 15:34:01 -05:00
return dir , nil
2013-03-18 03:15:35 -04:00
}
2015-01-27 21:10:28 -05:00
func ( graph * Graph ) newTempFile ( ) ( * os . File , error ) {
2015-06-05 18:31:10 -04:00
tmp , err := graph . mktemp ( "" )
2015-01-27 21:10:28 -05:00
if err != nil {
return nil , err
}
return ioutil . TempFile ( tmp , "" )
}
2015-03-30 14:19:12 -04:00
func bufferToFile ( f * os . File , src io . Reader ) ( int64 , digest . Digest , error ) {
var (
h = sha256 . New ( )
w = gzip . NewWriter ( io . MultiWriter ( f , h ) )
)
_ , err := io . Copy ( w , src )
w . Close ( )
2015-01-27 21:10:28 -05:00
if err != nil {
2015-03-30 14:19:12 -04:00
return 0 , "" , err
2015-01-27 21:10:28 -05:00
}
2015-03-30 14:19:12 -04:00
n , err := f . Seek ( 0 , os . SEEK_CUR )
if err != nil {
return 0 , "" , err
2015-01-27 21:10:28 -05:00
}
if _ , err := f . Seek ( 0 , 0 ) ; err != nil {
2015-03-30 14:19:12 -04:00
return 0 , "" , err
2015-01-27 21:10:28 -05:00
}
2015-03-30 14:19:12 -04:00
return n , digest . NewDigest ( "sha256" , h ) , nil
2015-01-27 21:10:28 -05:00
}
2013-03-30 00:13:59 -04:00
// Delete atomically removes an image from the graph.
2013-04-01 19:04:44 -04:00
func ( graph * Graph ) Delete ( name string ) error {
id , err := graph . idIndex . Get ( name )
if err != nil {
return err
}
2015-06-05 18:31:10 -04:00
tmp , err := graph . mktemp ( "" )
2013-04-01 01:11:55 -04:00
graph . idIndex . Delete ( id )
2014-09-22 10:47:20 -04:00
if err == nil {
2015-06-05 18:31:10 -04:00
if err := os . Rename ( graph . imageRoot ( id ) , tmp ) ; err != nil {
2015-04-26 12:50:25 -04:00
// On err make tmp point to old dir and cleanup unused tmp dir
2014-09-22 10:47:20 -04:00
os . RemoveAll ( tmp )
2015-06-05 18:31:10 -04:00
tmp = graph . imageRoot ( id )
2014-09-22 10:47:20 -04:00
}
} else {
// On err make tmp point to old dir for cleanup
2015-06-05 18:31:10 -04:00
tmp = graph . imageRoot ( id )
2013-03-18 03:15:35 -04:00
}
2013-11-07 15:34:01 -05:00
// Remove rootfs data from the driver
graph . driver . Remove ( id )
// Remove the trashed image directory
2013-04-04 01:14:28 -04:00
return os . RemoveAll ( tmp )
2013-03-18 03:15:35 -04:00
}
2013-03-30 03:22:24 -04:00
// Map returns a list of all images in the graph, addressable by ID.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) Map ( ) map [ string ] * image . Image {
images := make ( map [ string ] * image . Image )
graph . walkAll ( func ( image * image . Image ) {
2013-08-31 23:31:21 -04:00
images [ image . ID ] = image
} )
2015-06-19 11:01:39 -04:00
return images
2013-03-21 20:35:49 -04:00
}
2013-08-31 23:34:51 -04:00
// walkAll iterates over each image in the graph, and passes it to a handler.
2013-03-30 00:13:59 -04:00
// The walking order is undetermined.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) walkAll ( handler func ( * image . Image ) ) {
2015-06-19 11:01:39 -04:00
graph . idIndex . Iterate ( func ( id string ) {
if img , err := graph . Get ( id ) ; err != nil {
return
2013-03-23 20:03:30 -04:00
} else if handler != nil {
handler ( img )
}
2015-06-19 11:01:39 -04:00
} )
2013-03-23 20:03:30 -04:00
}
2013-03-30 00:13:59 -04:00
// ByParent returns a lookup table of images by their parent.
// If an image of id ID has 3 children images, then the value for key ID
// will be a list of 3 images.
// If an image has no children, it will not have an entry in the table.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) ByParent ( ) map [ string ] [ ] * image . Image {
byParent := make ( map [ string ] [ ] * image . Image )
graph . walkAll ( func ( img * image . Image ) {
2014-03-07 20:36:47 -05:00
parent , err := graph . Get ( img . Parent )
2013-03-23 20:03:30 -04:00
if err != nil {
return
}
2013-06-04 14:00:22 -04:00
if children , exists := byParent [ parent . ID ] ; exists {
2014-03-07 20:36:47 -05:00
byParent [ parent . ID ] = append ( children , img )
2013-08-11 03:37:16 -04:00
} else {
2015-07-20 13:57:15 -04:00
byParent [ parent . ID ] = [ ] * image . Image { img }
2013-03-18 03:15:35 -04:00
}
2013-03-23 20:03:30 -04:00
} )
2015-06-19 11:01:39 -04:00
return byParent
2013-03-23 20:03:30 -04:00
}
2015-07-29 19:45:47 -04:00
// Retain keeps the images and layers that are in the pulling chain so that
// they are not deleted. If not retained, they may be deleted by rmi.
2015-04-07 22:29:29 -04:00
func ( graph * Graph ) Retain ( sessionID string , layerIDs ... string ) {
graph . retained . Add ( sessionID , layerIDs )
}
2015-07-29 19:45:47 -04:00
// Release removes the referenced image ID from the provided set of layers.
2015-04-07 22:29:29 -04:00
func ( graph * Graph ) Release ( sessionID string , layerIDs ... string ) {
graph . retained . Delete ( sessionID , layerIDs )
}
2013-03-30 00:13:59 -04:00
// Heads returns all heads in the graph, keyed by id.
// A head is an image which is not the parent of another image in the graph.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) Heads ( ) map [ string ] * image . Image {
heads := make ( map [ string ] * image . Image )
2015-06-19 11:01:39 -04:00
byParent := graph . ByParent ( )
2015-07-20 13:57:15 -04:00
graph . walkAll ( func ( image * image . Image ) {
2013-03-23 20:03:30 -04:00
// If it's not in the byParent lookup table, then
// it's not a parent -> so it's a head!
2013-06-04 14:00:22 -04:00
if _ , exists := byParent [ image . ID ] ; ! exists {
heads [ image . ID ] = image
2013-03-23 20:03:30 -04:00
}
} )
2015-06-19 11:01:39 -04:00
return heads
2013-03-18 03:15:35 -04:00
}
2015-06-05 18:31:10 -04:00
func ( graph * Graph ) imageRoot ( id string ) string {
return filepath . Join ( graph . root , id )
2013-03-18 03:15:35 -04:00
}
2013-11-25 23:04:57 -05:00
2015-06-05 18:31:10 -04:00
// loadImage fetches the image with the given id from the graph.
2015-07-20 13:57:15 -04:00
func ( graph * Graph ) loadImage ( id string ) ( * image . Image , error ) {
2015-06-05 18:31:10 -04:00
root := graph . imageRoot ( id )
// Open the JSON file to decode by streaming
jsonSource , err := os . Open ( jsonPath ( root ) )
if err != nil {
return nil , err
}
defer jsonSource . Close ( )
2015-07-20 13:57:15 -04:00
img := & image . Image { }
2015-06-05 18:31:10 -04:00
dec := json . NewDecoder ( jsonSource )
// Decode the JSON data
if err := dec . Decode ( img ) ; err != nil {
return nil , err
}
if err := image . ValidateID ( img . ID ) ; err != nil {
return nil , err
}
2015-06-23 19:44:07 -04:00
if buf , err := ioutil . ReadFile ( filepath . Join ( root , layersizeFileName ) ) ; err != nil {
2015-06-05 18:31:10 -04:00
if ! os . IsNotExist ( err ) {
return nil , err
}
// If the layersize file does not exist then set the size to a negative number
// because a layer size of 0 (zero) is valid
img . Size = - 1
} else {
// Using Atoi here instead would temporarily convert the size to a machine
// dependent integer type, which causes images larger than 2^31 bytes to
// display negative sizes on 32-bit machines:
size , err := strconv . ParseInt ( string ( buf ) , 10 , 64 )
if err != nil {
return nil , err
}
img . Size = int64 ( size )
}
return img , nil
}
// saveSize stores the `size` in the provided graph `img` directory `root`.
2015-07-23 17:19:58 -04:00
func ( graph * Graph ) saveSize ( root string , size int64 ) error {
if err := ioutil . WriteFile ( filepath . Join ( root , layersizeFileName ) , [ ] byte ( strconv . FormatInt ( size , 10 ) ) , 0600 ) ; err != nil {
2015-06-23 19:44:07 -04:00
return fmt . Errorf ( "Error storing image size in %s/%s: %s" , root , layersizeFileName , err )
2015-06-05 18:31:10 -04:00
}
return nil
}
2015-06-05 21:07:41 -04:00
// SetDigest sets the digest for the image layer to the provided value.
func ( graph * Graph ) SetDigest ( id string , dgst digest . Digest ) error {
2015-06-05 18:31:10 -04:00
root := graph . imageRoot ( id )
2015-06-23 19:44:07 -04:00
if err := ioutil . WriteFile ( filepath . Join ( root , digestFileName ) , [ ] byte ( dgst . String ( ) ) , 0600 ) ; err != nil {
return fmt . Errorf ( "Error storing digest in %s/%s: %s" , root , digestFileName , err )
2015-06-05 18:31:10 -04:00
}
return nil
}
2015-06-05 21:07:41 -04:00
// GetDigest gets the digest for the provide image layer id.
func ( graph * Graph ) GetDigest ( id string ) ( digest . Digest , error ) {
2015-06-05 18:31:10 -04:00
root := graph . imageRoot ( id )
2015-06-23 19:44:07 -04:00
cs , err := ioutil . ReadFile ( filepath . Join ( root , digestFileName ) )
2015-06-05 18:31:10 -04:00
if err != nil {
if os . IsNotExist ( err ) {
2015-06-05 21:07:41 -04:00
return "" , ErrDigestNotSet
2015-06-05 18:31:10 -04:00
}
return "" , err
}
2015-06-05 21:07:41 -04:00
return digest . ParseDigest ( string ( cs ) )
2015-06-05 18:31:10 -04:00
}
// RawJSON returns the JSON representation for an image as a byte array.
func ( graph * Graph ) RawJSON ( id string ) ( [ ] byte , error ) {
root := graph . imageRoot ( id )
buf , err := ioutil . ReadFile ( jsonPath ( root ) )
if err != nil {
return nil , fmt . Errorf ( "Failed to read json for image %s: %s" , id , err )
}
return buf , nil
}
func jsonPath ( root string ) string {
2015-06-23 19:44:07 -04:00
return filepath . Join ( root , jsonFileName )
2013-03-18 03:15:35 -04:00
}
2015-07-20 21:32:14 -04:00
2015-08-03 21:52:54 -04:00
func ( graph * Graph ) disassembleAndApplyTarLayer ( img * image . Image , layerData archive . Reader , root string ) error {
2015-07-20 21:32:14 -04:00
// this is saving the tar-split metadata
mf , err := os . OpenFile ( filepath . Join ( root , tarDataFileName ) , os . O_CREATE | os . O_WRONLY | os . O_TRUNC , os . FileMode ( 0600 ) )
if err != nil {
return err
}
mfz := gzip . NewWriter ( mf )
metaPacker := storage . NewJSONPacker ( mfz )
defer mf . Close ( )
defer mfz . Close ( )
inflatedLayerData , err := archive . DecompressStream ( layerData )
if err != nil {
return err
}
// we're passing nil here for the file putter, because the ApplyDiff will
// handle the extraction of the archive
rdr , err := asm . NewInputTarStream ( inflatedLayerData , metaPacker , nil )
if err != nil {
return err
}
2015-08-03 21:52:54 -04:00
if img . Size , err = graph . driver . ApplyDiff ( img . ID , img . Parent , archive . Reader ( rdr ) ) ; err != nil {
2015-07-20 21:32:14 -04:00
return err
}
return nil
}
func ( graph * Graph ) assembleTarLayer ( img * image . Image ) ( archive . Archive , error ) {
root := graph . imageRoot ( img . ID )
mFileName := filepath . Join ( root , tarDataFileName )
mf , err := os . Open ( mFileName )
if err != nil {
if ! os . IsNotExist ( err ) {
logrus . Errorf ( "failed to open %q: %s" , mFileName , err )
}
return nil , err
}
pR , pW := io . Pipe ( )
// this will need to be in a goroutine, as we are returning the stream of a
// tar archive, but can not close the metadata reader early (when this
// function returns)...
go func ( ) {
defer mf . Close ( )
// let's reassemble!
logrus . Debugf ( "[graph] TarLayer with reassembly: %s" , img . ID )
mfz , err := gzip . NewReader ( mf )
if err != nil {
pW . CloseWithError ( fmt . Errorf ( "[graph] error with %s: %s" , mFileName , err ) )
return
}
defer mfz . Close ( )
// get our relative path to the container
fsLayer , err := graph . driver . Get ( img . ID , "" )
if err != nil {
pW . CloseWithError ( err )
return
}
defer graph . driver . Put ( img . ID )
metaUnpacker := storage . NewJSONUnpacker ( mfz )
fileGetter := storage . NewPathFileGetter ( fsLayer )
logrus . Debugf ( "[graph] %s is at %q" , img . ID , fsLayer )
ots := asm . NewOutputTarStream ( fileGetter , metaUnpacker )
defer ots . Close ( )
if _ , err := io . Copy ( pW , ots ) ; err != nil {
pW . CloseWithError ( err )
return
}
pW . Close ( )
} ( )
return pR , nil
}