2016-11-17 18:28:05 +00:00
|
|
|
package ocischema
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/docker/distribution"
|
2018-03-14 20:55:45 +00:00
|
|
|
"github.com/docker/distribution/manifest"
|
2021-10-21 22:09:39 +00:00
|
|
|
"github.com/docker/distribution/manifest/manifestlist"
|
|
|
|
|
2019-10-09 12:02:21 +00:00
|
|
|
v1 "github.com/opencontainers/image-spec/specs-go/v1"
|
2016-11-17 18:28:05 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var expectedManifestSerialization = []byte(`{
|
|
|
|
"schemaVersion": 2,
|
|
|
|
"mediaType": "application/vnd.oci.image.manifest.v1+json",
|
|
|
|
"config": {
|
|
|
|
"mediaType": "application/vnd.oci.image.config.v1+json",
|
|
|
|
"size": 985,
|
2017-07-21 01:44:02 +00:00
|
|
|
"digest": "sha256:1a9ec845ee94c202b2d5da74a24f0ed2058318bfa9879fa541efaecba272e86b",
|
|
|
|
"annotations": {
|
|
|
|
"apple": "orange"
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
},
|
|
|
|
"layers": [
|
|
|
|
{
|
|
|
|
"mediaType": "application/vnd.oci.image.layer.v1.tar+gzip",
|
|
|
|
"size": 153263,
|
2017-07-21 01:44:02 +00:00
|
|
|
"digest": "sha256:62d8908bee94c202b2d35224a221aaa2058318bfa9879fa541efaecba272331b",
|
|
|
|
"annotations": {
|
|
|
|
"lettuce": "wrap"
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
2017-07-31 23:34:11 +00:00
|
|
|
],
|
|
|
|
"annotations": {
|
|
|
|
"hot": "potato"
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
}`)
|
|
|
|
|
2018-03-14 20:55:45 +00:00
|
|
|
func makeTestManifest(mediaType string) Manifest {
|
|
|
|
return Manifest{
|
|
|
|
Versioned: manifest.Versioned{
|
|
|
|
SchemaVersion: 2,
|
|
|
|
MediaType: mediaType,
|
|
|
|
},
|
2016-11-17 18:28:05 +00:00
|
|
|
Config: distribution.Descriptor{
|
2017-07-21 01:44:02 +00:00
|
|
|
Digest: "sha256:1a9ec845ee94c202b2d5da74a24f0ed2058318bfa9879fa541efaecba272e86b",
|
|
|
|
Size: 985,
|
|
|
|
MediaType: v1.MediaTypeImageConfig,
|
|
|
|
Annotations: map[string]string{"apple": "orange"},
|
2016-11-17 18:28:05 +00:00
|
|
|
},
|
|
|
|
Layers: []distribution.Descriptor{
|
|
|
|
{
|
2017-07-21 01:44:02 +00:00
|
|
|
Digest: "sha256:62d8908bee94c202b2d35224a221aaa2058318bfa9879fa541efaecba272331b",
|
|
|
|
Size: 153263,
|
|
|
|
MediaType: v1.MediaTypeImageLayerGzip,
|
|
|
|
Annotations: map[string]string{"lettuce": "wrap"},
|
2016-11-17 18:28:05 +00:00
|
|
|
},
|
|
|
|
},
|
2017-07-31 23:34:11 +00:00
|
|
|
Annotations: map[string]string{"hot": "potato"},
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
2018-03-14 20:55:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func TestManifest(t *testing.T) {
|
|
|
|
manifest := makeTestManifest(v1.MediaTypeImageManifest)
|
2016-11-17 18:28:05 +00:00
|
|
|
|
|
|
|
deserialized, err := FromStruct(manifest)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error creating DeserializedManifest: %v", err)
|
|
|
|
}
|
|
|
|
|
2017-07-11 19:19:47 +00:00
|
|
|
mediaType, canonical, _ := deserialized.Payload()
|
2016-11-17 18:28:05 +00:00
|
|
|
|
2017-07-11 19:19:47 +00:00
|
|
|
if mediaType != v1.MediaTypeImageManifest {
|
2016-11-17 18:28:05 +00:00
|
|
|
t.Fatalf("unexpected media type: %s", mediaType)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that the canonical field is the same as json.MarshalIndent
|
|
|
|
// with these parameters.
|
|
|
|
p, err := json.MarshalIndent(&manifest, "", " ")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error marshaling manifest: %v", err)
|
|
|
|
}
|
|
|
|
if !bytes.Equal(p, canonical) {
|
|
|
|
t.Fatalf("manifest bytes not equal: %q != %q", string(canonical), string(p))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that canonical field matches expected value.
|
|
|
|
if !bytes.Equal(expectedManifestSerialization, canonical) {
|
|
|
|
t.Fatalf("manifest bytes not equal: %q != %q", string(canonical), string(expectedManifestSerialization))
|
|
|
|
}
|
|
|
|
|
|
|
|
var unmarshalled DeserializedManifest
|
|
|
|
if err := json.Unmarshal(deserialized.canonical, &unmarshalled); err != nil {
|
|
|
|
t.Fatalf("error unmarshaling manifest: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(&unmarshalled, deserialized) {
|
|
|
|
t.Fatalf("manifests are different after unmarshaling: %v != %v", unmarshalled, *deserialized)
|
|
|
|
}
|
2017-07-31 23:34:11 +00:00
|
|
|
if deserialized.Annotations["hot"] != "potato" {
|
|
|
|
t.Fatalf("unexpected annotation in manifest: %s", deserialized.Annotations["hot"])
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
|
|
|
|
target := deserialized.Target()
|
|
|
|
if target.Digest != "sha256:1a9ec845ee94c202b2d5da74a24f0ed2058318bfa9879fa541efaecba272e86b" {
|
|
|
|
t.Fatalf("unexpected digest in target: %s", target.Digest.String())
|
|
|
|
}
|
2017-07-11 19:19:47 +00:00
|
|
|
if target.MediaType != v1.MediaTypeImageConfig {
|
2016-11-17 18:28:05 +00:00
|
|
|
t.Fatalf("unexpected media type in target: %s", target.MediaType)
|
|
|
|
}
|
|
|
|
if target.Size != 985 {
|
|
|
|
t.Fatalf("unexpected size in target: %d", target.Size)
|
|
|
|
}
|
2017-07-21 01:44:02 +00:00
|
|
|
if target.Annotations["apple"] != "orange" {
|
|
|
|
t.Fatalf("unexpected annotation in target: %s", target.Annotations["apple"])
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
|
|
|
|
references := deserialized.References()
|
|
|
|
if len(references) != 2 {
|
|
|
|
t.Fatalf("unexpected number of references: %d", len(references))
|
|
|
|
}
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(references[0], target) {
|
|
|
|
t.Fatalf("first reference should be target: %v != %v", references[0], target)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test the second reference
|
|
|
|
if references[1].Digest != "sha256:62d8908bee94c202b2d35224a221aaa2058318bfa9879fa541efaecba272331b" {
|
|
|
|
t.Fatalf("unexpected digest in reference: %s", references[0].Digest.String())
|
|
|
|
}
|
2017-07-11 19:19:47 +00:00
|
|
|
if references[1].MediaType != v1.MediaTypeImageLayerGzip {
|
2016-11-17 18:28:05 +00:00
|
|
|
t.Fatalf("unexpected media type in reference: %s", references[0].MediaType)
|
|
|
|
}
|
|
|
|
if references[1].Size != 153263 {
|
|
|
|
t.Fatalf("unexpected size in reference: %d", references[0].Size)
|
|
|
|
}
|
2017-07-21 01:44:02 +00:00
|
|
|
if references[1].Annotations["lettuce"] != "wrap" {
|
|
|
|
t.Fatalf("unexpected annotation in reference: %s", references[1].Annotations["lettuce"])
|
|
|
|
}
|
2016-11-17 18:28:05 +00:00
|
|
|
}
|
2018-03-14 20:55:45 +00:00
|
|
|
|
|
|
|
func mediaTypeTest(t *testing.T, mediaType string, shouldError bool) {
|
|
|
|
manifest := makeTestManifest(mediaType)
|
|
|
|
|
|
|
|
deserialized, err := FromStruct(manifest)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error creating DeserializedManifest: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
unmarshalled, descriptor, err := distribution.UnmarshalManifest(
|
|
|
|
v1.MediaTypeImageManifest,
|
|
|
|
deserialized.canonical)
|
|
|
|
|
|
|
|
if shouldError {
|
|
|
|
if err == nil {
|
|
|
|
t.Fatalf("bad content type should have produced error")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("error unmarshaling manifest, %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
asManifest := unmarshalled.(*DeserializedManifest)
|
|
|
|
if asManifest.MediaType != mediaType {
|
|
|
|
t.Fatalf("Bad media type '%v' as unmarshalled", asManifest.MediaType)
|
|
|
|
}
|
|
|
|
|
|
|
|
if descriptor.MediaType != v1.MediaTypeImageManifest {
|
|
|
|
t.Fatalf("Bad media type '%v' for descriptor", descriptor.MediaType)
|
|
|
|
}
|
|
|
|
|
|
|
|
unmarshalledMediaType, _, _ := unmarshalled.Payload()
|
|
|
|
if unmarshalledMediaType != v1.MediaTypeImageManifest {
|
|
|
|
t.Fatalf("Bad media type '%v' for payload", unmarshalledMediaType)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestMediaTypes(t *testing.T) {
|
2018-03-14 21:22:36 +00:00
|
|
|
mediaTypeTest(t, "", false)
|
2018-03-14 20:55:45 +00:00
|
|
|
mediaTypeTest(t, v1.MediaTypeImageManifest, false)
|
|
|
|
mediaTypeTest(t, v1.MediaTypeImageManifest+"XXX", true)
|
|
|
|
}
|
2021-10-21 22:09:39 +00:00
|
|
|
|
|
|
|
func TestValidateManifest(t *testing.T) {
|
|
|
|
manifest := Manifest{
|
|
|
|
Config: distribution.Descriptor{Size: 1},
|
|
|
|
Layers: []distribution.Descriptor{{Size: 2}},
|
|
|
|
}
|
|
|
|
index := manifestlist.ManifestList{
|
|
|
|
Manifests: []manifestlist.ManifestDescriptor{
|
|
|
|
{Descriptor: distribution.Descriptor{Size: 3}},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
t.Run("valid", func(t *testing.T) {
|
|
|
|
b, err := json.Marshal(manifest)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("unexpected error marshaling manifest", err)
|
|
|
|
}
|
|
|
|
if err := validateManifest(b); err != nil {
|
|
|
|
t.Error("manifest should be valid", err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
t.Run("invalid", func(t *testing.T) {
|
|
|
|
b, err := json.Marshal(index)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal("unexpected error marshaling index", err)
|
|
|
|
}
|
|
|
|
if err := validateManifest(b); err == nil {
|
|
|
|
t.Error("index should not be valid")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|