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"
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"
2013-03-18 03:15:35 -04:00
)
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.
2013-03-18 03:15:35 -04:00
}
2015-06-11 14:29:29 -04:00
type Image struct {
ID string ` json:"id" `
Parent string ` json:"parent,omitempty" `
Comment string ` json:"comment,omitempty" `
Created time . Time ` json:"created" `
Container string ` json:"container,omitempty" `
ContainerConfig runconfig . Config ` json:"container_config,omitempty" `
DockerVersion string ` json:"docker_version,omitempty" `
Author string ` json:"author,omitempty" `
Config * runconfig . Config ` json:"config,omitempty" `
Architecture string ` json:"architecture,omitempty" `
OS string ` json:"os,omitempty" `
Size int64
graph Graph
}
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
2015-05-04 18:05:54 -04:00
if err := system . MkdirAll ( root , 0700 ) ; err != nil && ! os . IsExist ( err ) {
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-06-05 18:31:10 -04:00
root : abspath ,
2014-06-24 13:19:15 -04:00
idIndex : truncindex . NewTruncIndex ( [ ] string { } ) ,
2013-11-04 18:22:34 -05:00
driver : driver ,
2013-04-01 01:11:55 -04:00
}
if err := graph . restore ( ) ; err != nil {
return nil , err
}
return graph , nil
}
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
}
2013-03-26 08:28:17 -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
func ( graph * Graph ) IsNotExist ( err error , id string ) bool {
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-06-11 14:29:29 -04:00
func ( graph * Graph ) Get ( name string ) ( * 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-06-05 18:31:10 -04:00
if err := graph . saveSize ( graph . imageRoot ( id ) , int ( 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-06-11 14:29:29 -04:00
func ( graph * Graph ) Create ( layerData archive . ArchiveReader , containerID , containerImage , comment , author string , containerConfig , config * runconfig . Config ) ( * Image , error ) {
img := & 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-06-11 14:29:29 -04:00
func ( graph * Graph ) Register ( img * Image , layerData archive . ArchiveReader ) ( err error ) {
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 )
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.
// FIXME: this leaves a possible race condition.
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 ) {
dir := filepath . Join ( graph . root , "_tmp" , stringid . GenerateRandomID ( ) )
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-06-19 11:01:39 -04:00
func ( graph * Graph ) Map ( ) map [ string ] * Image {
2015-06-11 14:29:29 -04:00
images := make ( map [ string ] * Image )
2015-06-19 11:01:39 -04:00
graph . walkAll ( func ( 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-06-19 11:01:39 -04:00
func ( graph * Graph ) walkAll ( handler func ( * Image ) ) {
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-06-19 11:01:39 -04:00
func ( graph * Graph ) ByParent ( ) map [ string ] [ ] * Image {
2015-06-11 14:29:29 -04:00
byParent := make ( map [ string ] [ ] * Image )
2015-06-19 11:01:39 -04:00
graph . walkAll ( func ( img * 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-06-11 14:29:29 -04:00
byParent [ parent . ID ] = [ ] * 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
}
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-06-19 11:01:39 -04:00
func ( graph * Graph ) Heads ( ) map [ string ] * Image {
2015-06-11 14:29:29 -04:00
heads := make ( map [ string ] * Image )
2015-06-19 11:01:39 -04:00
byParent := graph . ByParent ( )
graph . walkAll ( func ( 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-06-11 14:29:29 -04:00
func ( graph * Graph ) loadImage ( id string ) ( * 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-06-11 14:29:29 -04:00
img := & 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
}
if buf , err := ioutil . ReadFile ( filepath . Join ( root , "layersize" ) ) ; err != nil {
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`.
func ( graph * Graph ) saveSize ( root string , size int ) error {
if err := ioutil . WriteFile ( filepath . Join ( root , "layersize" ) , [ ] byte ( strconv . Itoa ( size ) ) , 0600 ) ; err != nil {
return fmt . Errorf ( "Error storing image size in %s/layersize: %s" , root , err )
}
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-05 21:07:41 -04:00
if err := ioutil . WriteFile ( filepath . Join ( root , "checksum" ) , [ ] byte ( dgst . String ( ) ) , 0600 ) ; err != nil {
return fmt . Errorf ( "Error storing digest in %s/checksum: %s" , root , 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 )
cs , err := ioutil . ReadFile ( filepath . Join ( root , "checksum" ) )
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 {
return filepath . Join ( root , "json" )
2013-03-18 03:15:35 -04:00
}
2013-11-25 23:04:57 -05:00
2015-06-11 14:29:29 -04:00
// Build an Image object from raw json data
func NewImgJSON ( src [ ] byte ) ( * Image , error ) {
ret := & Image { }
// FIXME: Is there a cleaner way to "purify" the input json?
if err := json . Unmarshal ( src , ret ) ; err != nil {
return nil , err
}
return ret , nil
2013-11-25 23:04:57 -05:00
}