| package cache // import "github.com/docker/docker/image/cache" |
| |
| import ( |
| "runtime" |
| "testing" |
| |
| "github.com/docker/docker/api/types/container" |
| "github.com/docker/docker/api/types/strslice" |
| "github.com/docker/go-connections/nat" |
| ocispec "github.com/opencontainers/image-spec/specs-go/v1" |
| "gotest.tools/v3/assert" |
| is "gotest.tools/v3/assert/cmp" |
| ) |
| |
| // Just to make life easier |
| func newPortNoError(proto, port string) nat.Port { |
| p, _ := nat.NewPort(proto, port) |
| return p |
| } |
| |
| func TestCompare(t *testing.T) { |
| ports1 := make(nat.PortSet) |
| ports1[newPortNoError("tcp", "1111")] = struct{}{} |
| ports1[newPortNoError("tcp", "2222")] = struct{}{} |
| ports2 := make(nat.PortSet) |
| ports2[newPortNoError("tcp", "3333")] = struct{}{} |
| ports2[newPortNoError("tcp", "4444")] = struct{}{} |
| ports3 := make(nat.PortSet) |
| ports3[newPortNoError("tcp", "1111")] = struct{}{} |
| ports3[newPortNoError("tcp", "2222")] = struct{}{} |
| ports3[newPortNoError("tcp", "5555")] = struct{}{} |
| volumes1 := make(map[string]struct{}) |
| volumes1["/test1"] = struct{}{} |
| volumes2 := make(map[string]struct{}) |
| volumes2["/test2"] = struct{}{} |
| volumes3 := make(map[string]struct{}) |
| volumes3["/test1"] = struct{}{} |
| volumes3["/test3"] = struct{}{} |
| envs1 := []string{"ENV1=value1", "ENV2=value2"} |
| envs2 := []string{"ENV1=value1", "ENV3=value3"} |
| entrypoint1 := strslice.StrSlice{"/bin/sh", "-c"} |
| entrypoint2 := strslice.StrSlice{"/bin/sh", "-d"} |
| entrypoint3 := strslice.StrSlice{"/bin/sh", "-c", "echo"} |
| cmd1 := strslice.StrSlice{"/bin/sh", "-c"} |
| cmd2 := strslice.StrSlice{"/bin/sh", "-d"} |
| cmd3 := strslice.StrSlice{"/bin/sh", "-c", "echo"} |
| labels1 := map[string]string{"LABEL1": "value1", "LABEL2": "value2"} |
| labels2 := map[string]string{"LABEL1": "value1", "LABEL2": "value3"} |
| labels3 := map[string]string{"LABEL1": "value1", "LABEL2": "value2", "LABEL3": "value3"} |
| |
| sameConfigs := map[*container.Config]*container.Config{ |
| // Empty config |
| {}: {}, |
| // Does not compare hostname, domainname & image |
| { |
| Hostname: "host1", |
| Domainname: "domain1", |
| Image: "image1", |
| User: "user", |
| }: { |
| Hostname: "host2", |
| Domainname: "domain2", |
| Image: "image2", |
| User: "user", |
| }, |
| // only OpenStdin |
| {OpenStdin: false}: {OpenStdin: false}, |
| // only env |
| {Env: envs1}: {Env: envs1}, |
| // only cmd |
| {Cmd: cmd1}: {Cmd: cmd1}, |
| // only labels |
| {Labels: labels1}: {Labels: labels1}, |
| // only exposedPorts |
| {ExposedPorts: ports1}: {ExposedPorts: ports1}, |
| // only entrypoints |
| {Entrypoint: entrypoint1}: {Entrypoint: entrypoint1}, |
| // only volumes |
| {Volumes: volumes1}: {Volumes: volumes1}, |
| } |
| differentConfigs := map[*container.Config]*container.Config{ |
| nil: nil, |
| { |
| Hostname: "host1", |
| Domainname: "domain1", |
| Image: "image1", |
| User: "user1", |
| }: { |
| Hostname: "host1", |
| Domainname: "domain1", |
| Image: "image1", |
| User: "user2", |
| }, |
| // only OpenStdin |
| {OpenStdin: false}: {OpenStdin: true}, |
| {OpenStdin: true}: {OpenStdin: false}, |
| // only env |
| {Env: envs1}: {Env: envs2}, |
| // only cmd |
| {Cmd: cmd1}: {Cmd: cmd2}, |
| // not the same number of parts |
| {Cmd: cmd1}: {Cmd: cmd3}, |
| // only labels |
| {Labels: labels1}: {Labels: labels2}, |
| // not the same number of labels |
| {Labels: labels1}: {Labels: labels3}, |
| // only exposedPorts |
| {ExposedPorts: ports1}: {ExposedPorts: ports2}, |
| // not the same number of ports |
| {ExposedPorts: ports1}: {ExposedPorts: ports3}, |
| // only entrypoints |
| {Entrypoint: entrypoint1}: {Entrypoint: entrypoint2}, |
| // not the same number of parts |
| {Entrypoint: entrypoint1}: {Entrypoint: entrypoint3}, |
| // only volumes |
| {Volumes: volumes1}: {Volumes: volumes2}, |
| // not the same number of labels |
| {Volumes: volumes1}: {Volumes: volumes3}, |
| } |
| for config1, config2 := range sameConfigs { |
| if !compare(config1, config2) { |
| t.Fatalf("Compare should be true for [%v] and [%v]", config1, config2) |
| } |
| } |
| for config1, config2 := range differentConfigs { |
| if compare(config1, config2) { |
| t.Fatalf("Compare should be false for [%v] and [%v]", config1, config2) |
| } |
| } |
| } |
| |
| func TestPlatformCompare(t *testing.T) { |
| for _, tc := range []struct { |
| name string |
| builder ocispec.Platform |
| image ocispec.Platform |
| expected bool |
| }{ |
| { |
| name: "same os and arch", |
| builder: ocispec.Platform{Architecture: "amd64", OS: runtime.GOOS}, |
| image: ocispec.Platform{Architecture: "amd64", OS: runtime.GOOS}, |
| expected: true, |
| }, |
| { |
| name: "same os different arch", |
| builder: ocispec.Platform{Architecture: "amd64", OS: runtime.GOOS}, |
| image: ocispec.Platform{Architecture: "arm64", OS: runtime.GOOS}, |
| expected: false, |
| }, |
| { |
| name: "same os smaller host variant", |
| builder: ocispec.Platform{Variant: "v7", Architecture: "arm", OS: runtime.GOOS}, |
| image: ocispec.Platform{Variant: "v8", Architecture: "arm", OS: runtime.GOOS}, |
| expected: false, |
| }, |
| { |
| name: "same os higher host variant", |
| builder: ocispec.Platform{Variant: "v8", Architecture: "arm", OS: runtime.GOOS}, |
| image: ocispec.Platform{Variant: "v7", Architecture: "arm", OS: runtime.GOOS}, |
| expected: true, |
| }, |
| { |
| // Test for https://github.com/moby/moby/issues/47307 |
| name: "different build and revision", |
| builder: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.22621"}, |
| image: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| expected: true, |
| }, |
| { |
| name: "different revision", |
| builder: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.1234"}, |
| image: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| expected: true, |
| }, |
| { |
| name: "different major", |
| builder: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "11.0.17763.5329"}, |
| image: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| expected: false, |
| }, |
| { |
| name: "different minor same osver", |
| builder: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| image: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.1.17763.5329"}, |
| expected: false, |
| }, |
| { |
| name: "different arch same osver", |
| builder: ocispec.Platform{Architecture: "arm64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| image: ocispec.Platform{Architecture: "amd64", OS: "windows", OSVersion: "10.0.17763.5329"}, |
| expected: false, |
| }, |
| } { |
| tc := tc |
| // OSVersion comparison is only performed by containerd platform |
| // matcher if built on Windows. |
| if (tc.image.OSVersion != "" || tc.builder.OSVersion != "") && runtime.GOOS != "windows" { |
| continue |
| } |
| |
| t.Run(tc.name, func(t *testing.T) { |
| assert.Check(t, is.Equal(comparePlatform(tc.builder, tc.image), tc.expected)) |
| }) |
| } |
| } |