github.com/hustcat/docker@v1.3.3-0.20160314103604-901c67a8eeab/migrate/v1/migratev1_test.go (about)

     1  package v1
     2  
     3  import (
     4  	"crypto/rand"
     5  	"encoding/hex"
     6  	"encoding/json"
     7  	"fmt"
     8  	"io"
     9  	"io/ioutil"
    10  	"os"
    11  	"path/filepath"
    12  	"reflect"
    13  	"runtime"
    14  	"testing"
    15  
    16  	"github.com/docker/distribution/digest"
    17  	"github.com/docker/docker/distribution/metadata"
    18  	"github.com/docker/docker/image"
    19  	"github.com/docker/docker/layer"
    20  	"github.com/docker/docker/reference"
    21  )
    22  
    23  func TestMigrateRefs(t *testing.T) {
    24  	tmpdir, err := ioutil.TempDir("", "migrate-tags")
    25  	if err != nil {
    26  		t.Fatal(err)
    27  	}
    28  	defer os.RemoveAll(tmpdir)
    29  
    30  	ioutil.WriteFile(filepath.Join(tmpdir, "repositories-generic"), []byte(`{"Repositories":{"busybox":{"latest":"b3ca410aa2c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108","sha256:16a2a52884c2a9481ed267c2d46483eac7693b813a63132368ab098a71303f8a":"b3ca410aa2c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108"},"registry":{"2":"5d165b8e4b203685301c815e95663231691d383fd5e3d3185d1ce3f8dddead3d","latest":"8d5547a9f329b1d3f93198cd661fb5117e5a96b721c5cf9a2c389e7dd4877128"}}}`), 0600)
    31  
    32  	ta := &mockTagAdder{}
    33  	err = migrateRefs(tmpdir, "generic", ta, map[string]image.ID{
    34  		"5d165b8e4b203685301c815e95663231691d383fd5e3d3185d1ce3f8dddead3d": image.ID("sha256:2c26b46b68ffc68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae"),
    35  		"b3ca410aa2c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108": image.ID("sha256:fcde2b2edba56bf408601fb721fe9b5c338d10ee429ea04fae5511b68fbf8fb9"),
    36  		"abcdef3434c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108": image.ID("sha256:56434342345ae68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae"),
    37  	})
    38  	if err != nil {
    39  		t.Fatal(err)
    40  	}
    41  
    42  	expected := map[string]string{
    43  		"busybox:latest": "sha256:fcde2b2edba56bf408601fb721fe9b5c338d10ee429ea04fae5511b68fbf8fb9",
    44  		"busybox@sha256:16a2a52884c2a9481ed267c2d46483eac7693b813a63132368ab098a71303f8a": "sha256:fcde2b2edba56bf408601fb721fe9b5c338d10ee429ea04fae5511b68fbf8fb9",
    45  		"registry:2": "sha256:2c26b46b68ffc68ff99b453c1d30413413422d706483bfa0f98a5e886266e7ae",
    46  	}
    47  
    48  	if !reflect.DeepEqual(expected, ta.refs) {
    49  		t.Fatalf("Invalid migrated tags: expected %q, got %q", expected, ta.refs)
    50  	}
    51  
    52  	// second migration is no-op
    53  	ioutil.WriteFile(filepath.Join(tmpdir, "repositories-generic"), []byte(`{"Repositories":{"busybox":{"latest":"b3ca410aa2c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108"`), 0600)
    54  	err = migrateRefs(tmpdir, "generic", ta, map[string]image.ID{
    55  		"b3ca410aa2c115c05969a7b2c8cf8a9fcf62c1340ed6a601c9ee50df337ec108": image.ID("sha256:fcde2b2edba56bf408601fb721fe9b5c338d10ee429ea04fae5511b68fbf8fb9"),
    56  	})
    57  	if err != nil {
    58  		t.Fatal(err)
    59  	}
    60  	if !reflect.DeepEqual(expected, ta.refs) {
    61  		t.Fatalf("Invalid migrated tags: expected %q, got %q", expected, ta.refs)
    62  	}
    63  }
    64  
    65  func TestMigrateContainers(t *testing.T) {
    66  	// TODO Windows: Figure out why this is failing
    67  	if runtime.GOOS == "windows" {
    68  		t.Skip("Failing on Windows")
    69  	}
    70  	tmpdir, err := ioutil.TempDir("", "migrate-containers")
    71  	if err != nil {
    72  		t.Fatal(err)
    73  	}
    74  	defer os.RemoveAll(tmpdir)
    75  
    76  	err = addContainer(tmpdir, `{"State":{"Running":false,"Paused":false,"Restarting":false,"OOMKilled":false,"Dead":false,"Pid":0,"ExitCode":0,"Error":"","StartedAt":"2015-11-10T21:42:40.604267436Z","FinishedAt":"2015-11-10T21:42:41.869265487Z"},"ID":"f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c","Created":"2015-11-10T21:42:40.433831551Z","Path":"sh","Args":[],"Config":{"Hostname":"f780ee3f80e6","Domainname":"","User":"","AttachStdin":true,"AttachStdout":true,"AttachStderr":true,"Tty":true,"OpenStdin":true,"StdinOnce":true,"Env":null,"Cmd":["sh"],"Image":"busybox","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":{}},"Image":"2c5ac3f849df8627fcf2822727f87c57f38b7129d3604fbc11d861fe856ff093","NetworkSettings":{"Bridge":"","EndpointID":"","Gateway":"","GlobalIPv6Address":"","GlobalIPv6PrefixLen":0,"HairpinMode":false,"IPAddress":"","IPPrefixLen":0,"IPv6Gateway":"","LinkLocalIPv6Address":"","LinkLocalIPv6PrefixLen":0,"MacAddress":"","NetworkID":"","PortMapping":null,"Ports":null,"SandboxKey":"","SecondaryIPAddresses":null,"SecondaryIPv6Addresses":null},"ResolvConfPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/resolv.conf","HostnamePath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/hostname","HostsPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/hosts","LogPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c-json.log","Name":"/determined_euclid","Driver":"overlay","ExecDriver":"native-0.2","MountLabel":"","ProcessLabel":"","RestartCount":0,"UpdateDns":false,"HasBeenStartedBefore":false,"MountPoints":{},"Volumes":{},"VolumesRW":{},"AppArmorProfile":""}`)
    77  	if err != nil {
    78  		t.Fatal(err)
    79  	}
    80  
    81  	// container with invalid image
    82  	err = addContainer(tmpdir, `{"State":{"Running":false,"Paused":false,"Restarting":false,"OOMKilled":false,"Dead":false,"Pid":0,"ExitCode":0,"Error":"","StartedAt":"2015-11-10T21:42:40.604267436Z","FinishedAt":"2015-11-10T21:42:41.869265487Z"},"ID":"e780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c","Created":"2015-11-10T21:42:40.433831551Z","Path":"sh","Args":[],"Config":{"Hostname":"f780ee3f80e6","Domainname":"","User":"","AttachStdin":true,"AttachStdout":true,"AttachStderr":true,"Tty":true,"OpenStdin":true,"StdinOnce":true,"Env":null,"Cmd":["sh"],"Image":"busybox","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":{}},"Image":"4c5ac3f849df8627fcf2822727f87c57f38b7129d3604fbc11d861fe856ff093","NetworkSettings":{"Bridge":"","EndpointID":"","Gateway":"","GlobalIPv6Address":"","GlobalIPv6PrefixLen":0,"HairpinMode":false,"IPAddress":"","IPPrefixLen":0,"IPv6Gateway":"","LinkLocalIPv6Address":"","LinkLocalIPv6PrefixLen":0,"MacAddress":"","NetworkID":"","PortMapping":null,"Ports":null,"SandboxKey":"","SecondaryIPAddresses":null,"SecondaryIPv6Addresses":null},"ResolvConfPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/resolv.conf","HostnamePath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/hostname","HostsPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/hosts","LogPath":"/var/lib/docker/containers/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c/f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c-json.log","Name":"/determined_euclid","Driver":"overlay","ExecDriver":"native-0.2","MountLabel":"","ProcessLabel":"","RestartCount":0,"UpdateDns":false,"HasBeenStartedBefore":false,"MountPoints":{},"Volumes":{},"VolumesRW":{},"AppArmorProfile":""}`)
    83  	if err != nil {
    84  		t.Fatal(err)
    85  	}
    86  
    87  	ls := &mockMounter{}
    88  
    89  	ifs, err := image.NewFSStoreBackend(filepath.Join(tmpdir, "imagedb"))
    90  	if err != nil {
    91  		t.Fatal(err)
    92  	}
    93  
    94  	is, err := image.NewImageStore(ifs, ls)
    95  	if err != nil {
    96  		t.Fatal(err)
    97  	}
    98  
    99  	imgID, err := is.Create([]byte(`{"architecture":"amd64","config":{"AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Cmd":["sh"],"Entrypoint":null,"Env":null,"Hostname":"23304fc829f9","Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Labels":null,"OnBuild":null,"OpenStdin":false,"StdinOnce":false,"Tty":false,"Volumes":null,"WorkingDir":"","Domainname":"","User":""},"container":"349b014153779e30093d94f6df2a43c7a0a164e05aa207389917b540add39b51","container_config":{"AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Cmd":["/bin/sh","-c","#(nop) CMD [\"sh\"]"],"Entrypoint":null,"Env":null,"Hostname":"23304fc829f9","Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Labels":null,"OnBuild":null,"OpenStdin":false,"StdinOnce":false,"Tty":false,"Volumes":null,"WorkingDir":"","Domainname":"","User":""},"created":"2015-10-31T22:22:55.613815829Z","docker_version":"1.8.2","history":[{"created":"2015-10-31T22:22:54.690851953Z","created_by":"/bin/sh -c #(nop) ADD file:a3bc1e842b69636f9df5256c49c5374fb4eef1e281fe3f282c65fb853ee171c5 in /"},{"created":"2015-10-31T22:22:55.613815829Z","created_by":"/bin/sh -c #(nop) CMD [\"sh\"]"}],"os":"linux","rootfs":{"type":"layers","diff_ids":["sha256:c6f988f4874bb0add23a778f753c65efe992244e148a1d2ec2a8b664fb66bbd1","sha256:5f70bf18a086007016e948b04aed3b82103a36bea41755b6cddfaf10ace3c6ef"]}}`))
   100  	if err != nil {
   101  		t.Fatal(err)
   102  	}
   103  
   104  	err = migrateContainers(tmpdir, ls, is, map[string]image.ID{
   105  		"2c5ac3f849df8627fcf2822727f87c57f38b7129d3604fbc11d861fe856ff093": imgID,
   106  	})
   107  	if err != nil {
   108  		t.Fatal(err)
   109  	}
   110  
   111  	expected := []mountInfo{{
   112  		"f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c",
   113  		"f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c",
   114  		"sha256:c3191d32a37d7159b2e30830937d2e30268ad6c375a773a8994911a3aba9b93f",
   115  	}}
   116  	if !reflect.DeepEqual(expected, ls.mounts) {
   117  		t.Fatalf("invalid mounts: expected %q, got %q", expected, ls.mounts)
   118  	}
   119  
   120  	if actual, expected := ls.count, 0; actual != expected {
   121  		t.Fatalf("invalid active mounts: expected %d, got %d", expected, actual)
   122  	}
   123  
   124  	config2, err := ioutil.ReadFile(filepath.Join(tmpdir, "containers", "f780ee3f80e66e9b432a57049597118a66aab8932be88e5628d4c824edbee37c", "config.v2.json"))
   125  	if err != nil {
   126  		t.Fatal(err)
   127  	}
   128  	var config struct{ Image string }
   129  	err = json.Unmarshal(config2, &config)
   130  	if err != nil {
   131  		t.Fatal(err)
   132  	}
   133  
   134  	if actual, expected := config.Image, string(imgID); actual != expected {
   135  		t.Fatalf("invalid image pointer in migrated config: expected %q, got %q", expected, actual)
   136  	}
   137  
   138  }
   139  
   140  func TestMigrateImages(t *testing.T) {
   141  	// TODO Windows: Figure out why this is failing
   142  	if runtime.GOOS == "windows" {
   143  		t.Skip("Failing on Windows")
   144  	}
   145  	tmpdir, err := ioutil.TempDir("", "migrate-images")
   146  	if err != nil {
   147  		t.Fatal(err)
   148  	}
   149  	defer os.RemoveAll(tmpdir)
   150  
   151  	// busybox from 1.9
   152  	id1, err := addImage(tmpdir, `{"architecture":"amd64","config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":null,"Image":"","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":null},"container":"23304fc829f9b9349416f6eb1afec162907eba3a328f51d53a17f8986f865d65","container_config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["/bin/sh","-c","#(nop) ADD file:a3bc1e842b69636f9df5256c49c5374fb4eef1e281fe3f282c65fb853ee171c5 in /"],"Image":"","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":null},"created":"2015-10-31T22:22:54.690851953Z","docker_version":"1.8.2","layer_id":"sha256:55dc925c23d1ed82551fd018c27ac3ee731377b6bad3963a2a4e76e753d70e57","os":"linux"}`, "", "")
   153  	if err != nil {
   154  		t.Fatal(err)
   155  	}
   156  
   157  	id2, err := addImage(tmpdir, `{"architecture":"amd64","config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["sh"],"Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":null},"container":"349b014153779e30093d94f6df2a43c7a0a164e05aa207389917b540add39b51","container_config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["/bin/sh","-c","#(nop) CMD [\"sh\"]"],"Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Volumes":null,"WorkingDir":"","Entrypoint":null,"OnBuild":null,"Labels":null},"created":"2015-10-31T22:22:55.613815829Z","docker_version":"1.8.2","layer_id":"sha256:a3ed95caeb02ffe68cdd9fd84406680ae93d633cb16422d00e8a7c22955b46d4","os":"linux","parent_id":"sha256:039b63dd2cbaa10d6015ea574392530571ed8d7b174090f032211285a71881d0"}`, id1, "")
   158  	if err != nil {
   159  		t.Fatal(err)
   160  	}
   161  
   162  	ls := &mockRegistrar{}
   163  
   164  	ifs, err := image.NewFSStoreBackend(filepath.Join(tmpdir, "imagedb"))
   165  	if err != nil {
   166  		t.Fatal(err)
   167  	}
   168  
   169  	is, err := image.NewImageStore(ifs, ls)
   170  	if err != nil {
   171  		t.Fatal(err)
   172  	}
   173  
   174  	ms, err := metadata.NewFSMetadataStore(filepath.Join(tmpdir, "distribution"))
   175  	if err != nil {
   176  		t.Fatal(err)
   177  	}
   178  	mappings := make(map[string]image.ID)
   179  
   180  	err = migrateImages(tmpdir, ls, is, ms, mappings)
   181  	if err != nil {
   182  		t.Fatal(err)
   183  	}
   184  
   185  	expected := map[string]image.ID{
   186  		id1: image.ID("sha256:ca406eaf9c26898414ff5b7b3a023c33310759d6203be0663dbf1b3a712f432d"),
   187  		id2: image.ID("sha256:a488bec94bb96b26a968f913d25ef7d8d204d727ca328b52b4b059c7d03260b6"),
   188  	}
   189  
   190  	if !reflect.DeepEqual(mappings, expected) {
   191  		t.Fatalf("invalid image mappings: expected %q, got %q", expected, mappings)
   192  	}
   193  
   194  	if actual, expected := ls.count, 2; actual != expected {
   195  		t.Fatalf("invalid register count: expected %q, got %q", expected, actual)
   196  	}
   197  	ls.count = 0
   198  
   199  	// next images are busybox from 1.8.2
   200  	_, err = addImage(tmpdir, `{"id":"17583c7dd0dae6244203b8029733bdb7d17fccbb2b5d93e2b24cf48b8bfd06e2","parent":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","created":"2015-10-31T22:22:55.613815829Z","container":"349b014153779e30093d94f6df2a43c7a0a164e05aa207389917b540add39b51","container_config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"ExposedPorts":null,"PublishService":"","Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["/bin/sh","-c","#(nop) CMD [\"sh\"]"],"Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Volumes":null,"VolumeDriver":"","WorkingDir":"","Entrypoint":null,"NetworkDisabled":false,"MacAddress":"","OnBuild":null,"Labels":null},"docker_version":"1.8.2","config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"ExposedPorts":null,"PublishService":"","Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["sh"],"Image":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","Volumes":null,"VolumeDriver":"","WorkingDir":"","Entrypoint":null,"NetworkDisabled":false,"MacAddress":"","OnBuild":null,"Labels":null},"architecture":"amd64","os":"linux","Size":0}`, "", "sha256:a3ed95caeb02ffe68cdd9fd84406680ae93d633cb16422d00e8a7c22955b46d4")
   201  	if err != nil {
   202  		t.Fatal(err)
   203  	}
   204  
   205  	_, err = addImage(tmpdir, `{"id":"d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498","created":"2015-10-31T22:22:54.690851953Z","container":"23304fc829f9b9349416f6eb1afec162907eba3a328f51d53a17f8986f865d65","container_config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"ExposedPorts":null,"PublishService":"","Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":["/bin/sh","-c","#(nop) ADD file:a3bc1e842b69636f9df5256c49c5374fb4eef1e281fe3f282c65fb853ee171c5 in /"],"Image":"","Volumes":null,"VolumeDriver":"","WorkingDir":"","Entrypoint":null,"NetworkDisabled":false,"MacAddress":"","OnBuild":null,"Labels":null},"docker_version":"1.8.2","config":{"Hostname":"23304fc829f9","Domainname":"","User":"","AttachStdin":false,"AttachStdout":false,"AttachStderr":false,"ExposedPorts":null,"PublishService":"","Tty":false,"OpenStdin":false,"StdinOnce":false,"Env":null,"Cmd":null,"Image":"","Volumes":null,"VolumeDriver":"","WorkingDir":"","Entrypoint":null,"NetworkDisabled":false,"MacAddress":"","OnBuild":null,"Labels":null},"architecture":"amd64","os":"linux","Size":1108935}`, "", "sha256:55dc925c23d1ed82551fd018c27ac3ee731377b6bad3963a2a4e76e753d70e57")
   206  	if err != nil {
   207  		t.Fatal(err)
   208  	}
   209  
   210  	err = migrateImages(tmpdir, ls, is, ms, mappings)
   211  	if err != nil {
   212  		t.Fatal(err)
   213  	}
   214  
   215  	expected["d1592a710ac323612bd786fa8ac20727c58d8a67847e5a65177c594f43919498"] = image.ID("sha256:c091bb33854e57e6902b74c08719856d30b5593c7db6143b2b48376b8a588395")
   216  	expected["17583c7dd0dae6244203b8029733bdb7d17fccbb2b5d93e2b24cf48b8bfd06e2"] = image.ID("sha256:d963020e755ff2715b936065949472c1f8a6300144b922992a1a421999e71f07")
   217  
   218  	if actual, expected := ls.count, 2; actual != expected {
   219  		t.Fatalf("invalid register count: expected %q, got %q", expected, actual)
   220  	}
   221  
   222  	v2MetadataService := metadata.NewV2MetadataService(ms)
   223  	receivedMetadata, err := v2MetadataService.GetMetadata(layer.EmptyLayer.DiffID())
   224  	if err != nil {
   225  		t.Fatal(err)
   226  	}
   227  
   228  	expectedMetadata := []metadata.V2Metadata{
   229  		{Digest: digest.Digest("sha256:55dc925c23d1ed82551fd018c27ac3ee731377b6bad3963a2a4e76e753d70e57")},
   230  		{Digest: digest.Digest("sha256:a3ed95caeb02ffe68cdd9fd84406680ae93d633cb16422d00e8a7c22955b46d4")},
   231  	}
   232  
   233  	if !reflect.DeepEqual(expectedMetadata, receivedMetadata) {
   234  		t.Fatalf("invalid metadata: expected %q, got %q", expectedMetadata, receivedMetadata)
   235  	}
   236  
   237  }
   238  
   239  func TestMigrateUnsupported(t *testing.T) {
   240  	tmpdir, err := ioutil.TempDir("", "migrate-empty")
   241  	if err != nil {
   242  		t.Fatal(err)
   243  	}
   244  	defer os.RemoveAll(tmpdir)
   245  
   246  	err = os.MkdirAll(filepath.Join(tmpdir, "graph"), 0700)
   247  	if err != nil {
   248  		t.Fatal(err)
   249  	}
   250  
   251  	err = Migrate(tmpdir, "generic", nil, nil, nil, nil)
   252  	if err != errUnsupported {
   253  		t.Fatalf("expected unsupported error, got %q", err)
   254  	}
   255  }
   256  
   257  func TestMigrateEmptyDir(t *testing.T) {
   258  	tmpdir, err := ioutil.TempDir("", "migrate-empty")
   259  	if err != nil {
   260  		t.Fatal(err)
   261  	}
   262  	defer os.RemoveAll(tmpdir)
   263  
   264  	err = Migrate(tmpdir, "generic", nil, nil, nil, nil)
   265  	if err != nil {
   266  		t.Fatal(err)
   267  	}
   268  }
   269  
   270  func addImage(dest, jsonConfig, parent, checksum string) (string, error) {
   271  	var config struct{ ID string }
   272  	if err := json.Unmarshal([]byte(jsonConfig), &config); err != nil {
   273  		return "", err
   274  	}
   275  	if config.ID == "" {
   276  		b := make([]byte, 32)
   277  		rand.Read(b)
   278  		config.ID = hex.EncodeToString(b)
   279  	}
   280  	contDir := filepath.Join(dest, "graph", config.ID)
   281  	if err := os.MkdirAll(contDir, 0700); err != nil {
   282  		return "", err
   283  	}
   284  	if err := ioutil.WriteFile(filepath.Join(contDir, "json"), []byte(jsonConfig), 0600); err != nil {
   285  		return "", err
   286  	}
   287  	if checksum != "" {
   288  		if err := ioutil.WriteFile(filepath.Join(contDir, "checksum"), []byte(checksum), 0600); err != nil {
   289  			return "", err
   290  		}
   291  	}
   292  	if err := ioutil.WriteFile(filepath.Join(contDir, ".migration-diffid"), []byte(layer.EmptyLayer.DiffID()), 0600); err != nil {
   293  		return "", err
   294  	}
   295  	if err := ioutil.WriteFile(filepath.Join(contDir, ".migration-size"), []byte("0"), 0600); err != nil {
   296  		return "", err
   297  	}
   298  	if parent != "" {
   299  		if err := ioutil.WriteFile(filepath.Join(contDir, "parent"), []byte(parent), 0600); err != nil {
   300  			return "", err
   301  		}
   302  	}
   303  	if checksum != "" {
   304  		if err := ioutil.WriteFile(filepath.Join(contDir, "checksum"), []byte(checksum), 0600); err != nil {
   305  			return "", err
   306  		}
   307  	}
   308  	return config.ID, nil
   309  }
   310  
   311  func addContainer(dest, jsonConfig string) error {
   312  	var config struct{ ID string }
   313  	if err := json.Unmarshal([]byte(jsonConfig), &config); err != nil {
   314  		return err
   315  	}
   316  	contDir := filepath.Join(dest, "containers", config.ID)
   317  	if err := os.MkdirAll(contDir, 0700); err != nil {
   318  		return err
   319  	}
   320  	if err := ioutil.WriteFile(filepath.Join(contDir, "config.json"), []byte(jsonConfig), 0600); err != nil {
   321  		return err
   322  	}
   323  	return nil
   324  }
   325  
   326  type mockTagAdder struct {
   327  	refs map[string]string
   328  }
   329  
   330  func (t *mockTagAdder) AddTag(ref reference.Named, id image.ID, force bool) error {
   331  	if t.refs == nil {
   332  		t.refs = make(map[string]string)
   333  	}
   334  	t.refs[ref.String()] = id.String()
   335  	return nil
   336  }
   337  func (t *mockTagAdder) AddDigest(ref reference.Canonical, id image.ID, force bool) error {
   338  	return t.AddTag(ref, id, force)
   339  }
   340  
   341  type mockRegistrar struct {
   342  	layers map[layer.ChainID]*mockLayer
   343  	count  int
   344  }
   345  
   346  func (r *mockRegistrar) RegisterByGraphID(graphID string, parent layer.ChainID, diffID layer.DiffID, tarDataFile string, size int64) (layer.Layer, error) {
   347  	r.count++
   348  	l := &mockLayer{}
   349  	if parent != "" {
   350  		p, exists := r.layers[parent]
   351  		if !exists {
   352  			return nil, fmt.Errorf("invalid parent %q", parent)
   353  		}
   354  		l.parent = p
   355  		l.diffIDs = append(l.diffIDs, p.diffIDs...)
   356  	}
   357  	l.diffIDs = append(l.diffIDs, diffID)
   358  	if r.layers == nil {
   359  		r.layers = make(map[layer.ChainID]*mockLayer)
   360  	}
   361  	r.layers[l.ChainID()] = l
   362  	return l, nil
   363  }
   364  func (r *mockRegistrar) Release(l layer.Layer) ([]layer.Metadata, error) {
   365  	return nil, nil
   366  }
   367  func (r *mockRegistrar) Get(layer.ChainID) (layer.Layer, error) {
   368  	return nil, nil
   369  }
   370  
   371  type mountInfo struct {
   372  	name, graphID, parent string
   373  }
   374  type mockMounter struct {
   375  	mounts []mountInfo
   376  	count  int
   377  }
   378  
   379  func (r *mockMounter) CreateRWLayerByGraphID(name string, graphID string, parent layer.ChainID) error {
   380  	r.mounts = append(r.mounts, mountInfo{name, graphID, string(parent)})
   381  	return nil
   382  }
   383  func (r *mockMounter) Unmount(string) error {
   384  	r.count--
   385  	return nil
   386  }
   387  func (r *mockMounter) Get(layer.ChainID) (layer.Layer, error) {
   388  	return nil, nil
   389  }
   390  
   391  func (r *mockMounter) Release(layer.Layer) ([]layer.Metadata, error) {
   392  	return nil, nil
   393  }
   394  
   395  type mockLayer struct {
   396  	diffIDs []layer.DiffID
   397  	parent  *mockLayer
   398  }
   399  
   400  func (l *mockLayer) TarStream() (io.ReadCloser, error) {
   401  	return nil, nil
   402  }
   403  
   404  func (l *mockLayer) ChainID() layer.ChainID {
   405  	return layer.CreateChainID(l.diffIDs)
   406  }
   407  
   408  func (l *mockLayer) DiffID() layer.DiffID {
   409  	return l.diffIDs[len(l.diffIDs)-1]
   410  }
   411  
   412  func (l *mockLayer) Parent() layer.Layer {
   413  	if l.parent == nil {
   414  		return nil
   415  	}
   416  	return l.parent
   417  }
   418  
   419  func (l *mockLayer) Size() (int64, error) {
   420  	return 0, nil
   421  }
   422  
   423  func (l *mockLayer) DiffSize() (int64, error) {
   424  	return 0, nil
   425  }
   426  
   427  func (l *mockLayer) Metadata() (map[string]string, error) {
   428  	return nil, nil
   429  }