diff --git a/.github/config/golangci.yaml b/.github/config/golangci.yaml index 8352fbc2a3..01770b48d1 100644 --- a/.github/config/golangci.yaml +++ b/.github/config/golangci.yaml @@ -173,4 +173,26 @@ issues: - path: ignore/.*\.go linters: - dupword + # Deprecated algorithms and fields for extra identity field defaulting + # TODO: To be removed once v1 + v2 are removed. + - path: "cmds/.*|api/.*" + linters: + - staticcheck + text: "SA1019: jsonv1.Algorithm is deprecated" + - path: "cmds/.*|api/.*" + linters: + - staticcheck + text: "SA1019: compdesc.JsonNormalisationV1 is deprecated" + - path: "cmds/.*|api/.*" + linters: + - staticcheck + text: "SA1019: jsonv2.Algorithm is deprecated" + - path: "cmds/.*|api/.*" + linters: + - staticcheck + text: "SA1019: compdesc.JsonNormalisationV2 is deprecated" + - path: "cmds/.*|api/.*" + linters: + - staticcheck + text: "SA1019: legacy.DefaultingOfVersionIntoExtraIdentity is deprecated" diff --git a/.github/config/wordlist.txt b/.github/config/wordlist.txt index d38b0d4469..55c8a10bd4 100644 --- a/.github/config/wordlist.txt +++ b/.github/config/wordlist.txt @@ -308,4 +308,5 @@ xml yaml yitsushi yml -yyyy \ No newline at end of file +yyyy +jsonNormalisation \ No newline at end of file diff --git a/.github/workflows/lint_and_test.yaml b/.github/workflows/lint_and_test.yaml index 20bfddb4a1..fea754026f 100644 --- a/.github/workflows/lint_and_test.yaml +++ b/.github/workflows/lint_and_test.yaml @@ -111,3 +111,10 @@ jobs: run: go install golang.org/x/tools/cmd/goimports@latest - name: Lint run: make check + + codespell: # call reusable workflow from central '.github' repo + uses: open-component-model/.github/.github/workflows/codespell.yml@main + secrets: inherit + with: + codespell-ignore: .github/config/wordlist.txt + codespell-files-glob: ./**/*.y*ml ./**/*.go diff --git a/api/ocm/compdesc/norm_test.go b/api/ocm/compdesc/norm_test.go index 8d3991f076..786f35ea77 100644 --- a/api/ocm/compdesc/norm_test.go +++ b/api/ocm/compdesc/norm_test.go @@ -171,7 +171,7 @@ var _ = Describe("Normalization", func() { Expect(err).To(Succeed()) }) - It("hashes first", func() { + It("normalizes v1", func() { n, err := compdesc.Normalize(cd1, compdesc.JsonNormalisationV1) Expect(err).To(Succeed()) Expect(string(n)).To(StringEqualTrimmedWithContext("[{\"component\":[{\"componentReferences\":[]},{\"name\":\"github.com/vasu1124/introspect\"},{\"provider\":\"internal\"},{\"resources\":[[{\"digest\":[{\"hashAlgorithm\":\"SHA-256\"},{\"normalisationAlgorithm\":\"ociArtifactDigest/v1\"},{\"value\":\"6a1c7637a528ab5957ab60edf73b5298a0a03de02a96be0313ee89b22544840c\"}]},{\"extraIdentity\":null},{\"labels\":[[{\"name\":\"label2\"},{\"signing\":true},{\"value\":\"bar\"}]]},{\"name\":\"introspect-image\"},{\"relation\":\"local\"},{\"type\":\"ociImage\"},{\"version\":\"1.0.0\"}],[{\"digest\":[{\"hashAlgorithm\":\"SHA-256\"},{\"normalisationAlgorithm\":\"genericBlobDigest/v1\"},{\"value\":\"d1187ac17793b2f5fa26175c21cabb6ce388871ae989e16ff9a38bd6b32507bf\"}]},{\"extraIdentity\":null},{\"name\":\"introspect-blueprint\"},{\"relation\":\"local\"},{\"type\":\"landscaper.gardener.cloud/blueprint\"},{\"version\":\"1.0.0\"}],[{\"digest\":[{\"hashAlgorithm\":\"SHA-256\"},{\"normalisationAlgorithm\":\"ociArtifactDigest/v1\"},{\"value\":\"6229be2be7e328f74ba595d93b814b590b1aa262a1b85e49cc1492795a9e564c\"}]},{\"extraIdentity\":null},{\"name\":\"introspect-helm\"},{\"relation\":\"external\"},{\"type\":\"helm\"},{\"version\":\"0.1.0\"}]]},{\"version\":\"1.0.0\"}]},{\"meta\":[{\"schemaVersion\":\"v2\"}]}]")) @@ -180,13 +180,22 @@ var _ = Describe("Normalization", func() { Expect(o).To(Equal(n)) }) - It("hashes v2", func() { + It("normalizes v2", func() { n, err := compdesc.Normalize(cd1, compdesc.JsonNormalisationV2) Expect(err).To(Succeed()) Expect(string(n)).To(Equal(`{"component":{"componentReferences":[],"name":"github.com/vasu1124/introspect","provider":{"name":"internal"},"resources":[{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6a1c7637a528ab5957ab60edf73b5298a0a03de02a96be0313ee89b22544840c"},"labels":[{"name":"label2","signing":true,"value":"bar"}],"name":"introspect-image","relation":"local","type":"ociImage","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"genericBlobDigest/v1","value":"d1187ac17793b2f5fa26175c21cabb6ce388871ae989e16ff9a38bd6b32507bf"},"name":"introspect-blueprint","relation":"local","type":"landscaper.gardener.cloud/blueprint","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6229be2be7e328f74ba595d93b814b590b1aa262a1b85e49cc1492795a9e564c"},"name":"introspect-helm","relation":"external","type":"helm","version":"0.1.0"}],"sources":[{"name":"introspect","type":"git","version":"1.0.0"}],"version":"1.0.0"}}`)) }) - It("hashes v1 with none access", func() { + It("normalises v3", func() { + n, err := compdesc.Normalize(cd1, compdesc.JsonNormalisationV3) + Expect(err).To(Succeed()) + Expect(string(n)).To(Equal(`{"component":{"componentReferences":[],"name":"github.com/vasu1124/introspect","provider":{"name":"internal"},"resources":[{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6a1c7637a528ab5957ab60edf73b5298a0a03de02a96be0313ee89b22544840c"},"labels":[{"name":"label2","signing":true,"value":"bar"}],"name":"introspect-image","relation":"local","type":"ociImage","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"genericBlobDigest/v1","value":"d1187ac17793b2f5fa26175c21cabb6ce388871ae989e16ff9a38bd6b32507bf"},"name":"introspect-blueprint","relation":"local","type":"landscaper.gardener.cloud/blueprint","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6229be2be7e328f74ba595d93b814b590b1aa262a1b85e49cc1492795a9e564c"},"name":"introspect-helm","relation":"external","type":"helm","version":"0.1.0"}],"sources":[{"name":"introspect","type":"git","version":"1.0.0"}],"version":"1.0.0"}}`)) + o, err := compdesc.Normalize(cd1, compdesc.JsonNormalisationV2) + Expect(err).To(Succeed()) + Expect(o).To(Equal(n)) + }) + + It("normalizes v1 with none access", func() { cd1.Resources = append(cd1.Resources, compdesc.Resource{ ResourceMeta: compdesc.ResourceMeta{ ElementMeta: compdesc.ElementMeta{ @@ -208,7 +217,7 @@ var _ = Describe("Normalization", func() { Expect(string(n)).To(StringEqualWithContext(`[{"component":[{"componentReferences":[]},{"name":"github.com/vasu1124/introspect"},{"provider":"internal"},{"resources":[[{"digest":[{"hashAlgorithm":"SHA-256"},{"normalisationAlgorithm":"ociArtifactDigest/v1"},{"value":"6a1c7637a528ab5957ab60edf73b5298a0a03de02a96be0313ee89b22544840c"}]},{"extraIdentity":null},{"labels":[[{"name":"label2"},{"signing":true},{"value":"bar"}]]},{"name":"introspect-image"},{"relation":"local"},{"type":"ociImage"},{"version":"1.0.0"}],[{"digest":[{"hashAlgorithm":"SHA-256"},{"normalisationAlgorithm":"genericBlobDigest/v1"},{"value":"d1187ac17793b2f5fa26175c21cabb6ce388871ae989e16ff9a38bd6b32507bf"}]},{"extraIdentity":null},{"name":"introspect-blueprint"},{"relation":"local"},{"type":"landscaper.gardener.cloud/blueprint"},{"version":"1.0.0"}],[{"digest":[{"hashAlgorithm":"SHA-256"},{"normalisationAlgorithm":"ociArtifactDigest/v1"},{"value":"6229be2be7e328f74ba595d93b814b590b1aa262a1b85e49cc1492795a9e564c"}]},{"extraIdentity":null},{"name":"introspect-helm"},{"relation":"external"},{"type":"helm"},{"version":"0.1.0"}],[{"extraIdentity":null},{"name":"none"},{"relation":"local"},{"type":"plainText"},{"version":"v1"}]]},{"version":"1.0.0"}]},{"meta":[{"schemaVersion":"v2"}]}]`)) }) - It("hashes v2 with none access", func() { + It("normalizes v2 with none access", func() { cd1.Resources = append(cd1.Resources, compdesc.Resource{ ResourceMeta: compdesc.ResourceMeta{ ElementMeta: compdesc.ElementMeta{ @@ -230,7 +239,7 @@ var _ = Describe("Normalization", func() { Expect(string(n)).To(Equal(`{"component":{"componentReferences":[],"name":"github.com/vasu1124/introspect","provider":{"name":"internal"},"resources":[{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6a1c7637a528ab5957ab60edf73b5298a0a03de02a96be0313ee89b22544840c"},"labels":[{"name":"label2","signing":true,"value":"bar"}],"name":"introspect-image","relation":"local","type":"ociImage","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"genericBlobDigest/v1","value":"d1187ac17793b2f5fa26175c21cabb6ce388871ae989e16ff9a38bd6b32507bf"},"name":"introspect-blueprint","relation":"local","type":"landscaper.gardener.cloud/blueprint","version":"1.0.0"},{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"ociArtifactDigest/v1","value":"6229be2be7e328f74ba595d93b814b590b1aa262a1b85e49cc1492795a9e564c"},"name":"introspect-helm","relation":"external","type":"helm","version":"0.1.0"},{"name":"none","relation":"local","type":"plainText","version":"v1"}],"sources":[{"name":"introspect","type":"git","version":"1.0.0"}],"version":"1.0.0"}}`)) }) - It("hashes v2 with complex provider", func() { + It("normalizes v2 with complex provider", func() { cd := cd1.Copy() cd.References = nil cd.Resources = nil @@ -248,7 +257,7 @@ var _ = Describe("Normalization", func() { Expect(string(n)).To(Equal(`{"component":{"componentReferences":[],"labels":[{"name":"non-volatile","signing":true,"value":"comp-value2"}],"name":"github.com/vasu1124/introspect","provider":{"labels":[{"name":"non-volatile","signing":true,"value":"prov-value2"}],"name":"internal"},"resources":[],"sources":[],"version":"1.0.0"}}`)) }) - It("hashes v1 with complex provider for CD/v2", func() { + It("normalizes v1 with complex provider for CD/v2", func() { cd := cd1.Copy() cd.References = nil cd.Resources = nil @@ -266,7 +275,7 @@ var _ = Describe("Normalization", func() { Expect(string(n)).To(StringEqualWithContext(`[{"component":[{"componentReferences":[]},{"labels":[[{"name":"non-volatile"},{"signing":true},{"value":"comp-value2"}]]},{"name":"github.com/vasu1124/introspect"},{"provider":[{"labels":[[{"name":"non-volatile"},{"signing":true},{"value":"prov-value2"}]]},{"name":"internal"}]},{"resources":[]},{"version":"1.0.0"}]},{"meta":[{"schemaVersion":"v2"}]}]`)) }) - It("hashes v1 with complex provider for CD/v3", func() { + It("normalizes v1 with complex provider for CD/v3", func() { cd := cd1.Copy() cd.Metadata.ConfiguredVersion = v3alpha1.SchemaVersion cd.References = nil @@ -284,4 +293,51 @@ var _ = Describe("Normalization", func() { Expect(string(n)).To(StringEqualWithContext(`[{"apiVersion":"ocm.software/v3alpha1"},{"kind":"ComponentVersion"},{"metadata":[{"labels":[[{"name":"non-volatile"},{"signing":true},{"value":"comp-value2"}]]},{"name":"github.com/vasu1124/introspect"},{"provider":[{"labels":[[{"name":"volatile"},{"value":"prov-value1"}],[{"name":"non-volatile"},{"signing":true},{"value":"prov-value2"}]]},{"name":"internal"}]},{"version":"1.0.0"}]},{"spec":[]}]`)) }) + + Context("normalization and legacy extra identity defaulting", func() { + var cd *compdesc.ComponentDescriptor + BeforeEach(func() { + cd = Must(compdesc.Decode([]byte(` + component: + version: 1.0.0 + componentReferences: [] + name: ocm.software/duplicate-resource/test + provider: internal + repositoryContexts: [] + resources: + - name: image + relation: local + type: ociImage + version: 1.0.0 + access: + imageReference: ghcr.io/bla:1.0.0 + type: ociRegistry + - name: image + relation: local + type: ociImage + version: 2.0.0 + access: + imageReference: ghcr.io/bla:2.0.0 + type: ociRegistry + sources: [] + meta: + schemaVersion: v2 +`))) + }) + It("normalizes v1 with extra identity defaulting", func() { + n := Must(compdesc.Normalize(cd, compdesc.JsonNormalisationV1)) + Expect(string(n)).To(StringEqualTrimmedWithContext("[{\"component\":[{\"componentReferences\":[]},{\"name\":\"ocm.software/duplicate-resource/test\"},{\"provider\":\"internal\"},{\"resources\":[[{\"extraIdentity\":[{\"version\":\"1.0.0\"}]},{\"name\":\"image\"},{\"relation\":\"local\"},{\"type\":\"ociImage\"},{\"version\":\"1.0.0\"}],[{\"extraIdentity\":null},{\"name\":\"image\"},{\"relation\":\"local\"},{\"type\":\"ociImage\"},{\"version\":\"2.0.0\"}]]},{\"version\":\"1.0.0\"}]},{\"meta\":[{\"schemaVersion\":\"v2\"}]}]")) + Expect(string(n)).To(ContainSubstring("\"extraIdentity\":[{\"version\":\"1.0.0\"}]"), "extra identity should have been defaulted, see api/ocm/compdesc/normalizations/legacy/DefaultingOfVersionIntoExtraIdentity") + }) + It("normalizes v2 with extra identity defaulting", func() { + n := Must(compdesc.Normalize(cd, compdesc.JsonNormalisationV2)) + Expect(string(n)).To(StringEqualTrimmedWithContext("{\"component\":{\"componentReferences\":[],\"name\":\"ocm.software/duplicate-resource/test\",\"provider\":{\"name\":\"internal\"},\"resources\":[{\"extraIdentity\":{\"version\":\"1.0.0\"},\"name\":\"image\",\"relation\":\"local\",\"type\":\"ociImage\",\"version\":\"1.0.0\"},{\"name\":\"image\",\"relation\":\"local\",\"type\":\"ociImage\",\"version\":\"2.0.0\"}],\"sources\":[],\"version\":\"1.0.0\"}}")) + Expect(string(n)).To(ContainSubstring("{\"extraIdentity\":{\"version\":\"1.0.0\"}"), "extra identity should have been defaulted, see api/ocm/compdesc/normalizations/legacy/DefaultingOfVersionIntoExtraIdentity") + }) + It("normalizes v3 without extra identity defaulting", func() { + n := Must(compdesc.Normalize(cd, compdesc.JsonNormalisationV3)) + Expect(string(n)).To(StringEqualTrimmedWithContext("{\"component\":{\"componentReferences\":[],\"name\":\"ocm.software/duplicate-resource/test\",\"provider\":{\"name\":\"internal\"},\"resources\":[{\"name\":\"image\",\"relation\":\"local\",\"type\":\"ociImage\",\"version\":\"1.0.0\"},{\"name\":\"image\",\"relation\":\"local\",\"type\":\"ociImage\",\"version\":\"2.0.0\"}],\"sources\":[],\"version\":\"1.0.0\"}}")) + Expect(string(n)).ToNot(ContainSubstring("{\"extraIdentity\":{\"version\":\"1.0.0\"}"), "extra identity should not have been defaulted") + }) + }) }) diff --git a/api/ocm/compdesc/normalization.go b/api/ocm/compdesc/normalization.go index 5084f1ca89..6e7687d5cb 100644 --- a/api/ocm/compdesc/normalization.go +++ b/api/ocm/compdesc/normalization.go @@ -11,8 +11,11 @@ import ( type NormalisationAlgorithm = string const ( + // Deprecated: use JsonNormalisationV3 instead JsonNormalisationV1 NormalisationAlgorithm = "jsonNormalisation/v1" + // Deprecated: use JsonNormalisationV3 instead JsonNormalisationV2 NormalisationAlgorithm = "jsonNormalisation/v2" + JsonNormalisationV3 NormalisationAlgorithm = "jsonNormalisation/v3" ) type Normalization interface { diff --git a/api/ocm/compdesc/normalizations/init.go b/api/ocm/compdesc/normalizations/init.go index 96dab531de..6b4eeefa3e 100644 --- a/api/ocm/compdesc/normalizations/init.go +++ b/api/ocm/compdesc/normalizations/init.go @@ -3,4 +3,5 @@ package normalizations import ( _ "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv1" _ "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv2" + _ "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv3" ) diff --git a/api/ocm/compdesc/normalizations/jsonv1/norm.go b/api/ocm/compdesc/normalizations/jsonv1/norm.go index 5a6f9bf141..ae24494fa2 100644 --- a/api/ocm/compdesc/normalizations/jsonv1/norm.go +++ b/api/ocm/compdesc/normalizations/jsonv1/norm.go @@ -8,9 +8,11 @@ import ( "github.com/mandelsoft/goutils/errors" "ocm.software/ocm/api/ocm/compdesc" + "ocm.software/ocm/api/ocm/compdesc/normalizations/legacy" "ocm.software/ocm/api/utils/errkind" ) +// Deprecated: use compdesc.JsonNormalisationV3 instead const Algorithm = compdesc.JsonNormalisationV1 func init() { @@ -20,11 +22,10 @@ func init() { type normalization struct{} func (m normalization) Normalize(cd *compdesc.ComponentDescriptor) ([]byte, error) { + legacy.DefaultingOfVersionIntoExtraIdentity(cd) cv := compdesc.DefaultSchemes[cd.SchemaVersion()] if cv == nil { - if cv == nil { - return nil, errors.ErrNotSupported(errkind.KIND_SCHEMAVERSION, cd.SchemaVersion()) - } + return nil, errors.ErrNotSupported(errkind.KIND_SCHEMAVERSION, cd.SchemaVersion()) } v, err := cv.ConvertFrom(cd) if err != nil { diff --git a/api/ocm/compdesc/normalizations/jsonv2/norm.go b/api/ocm/compdesc/normalizations/jsonv2/norm.go index 1fcb3a98e8..9fb7b3f284 100644 --- a/api/ocm/compdesc/normalizations/jsonv2/norm.go +++ b/api/ocm/compdesc/normalizations/jsonv2/norm.go @@ -10,11 +10,13 @@ package jsonv2 import ( "ocm.software/ocm/api/ocm/compdesc" + "ocm.software/ocm/api/ocm/compdesc/normalizations/legacy" "ocm.software/ocm/api/ocm/compdesc/normalizations/rules" "ocm.software/ocm/api/tech/signing" "ocm.software/ocm/api/tech/signing/norm/jcs" ) +// Deprecated: use compdesc.JsonNormalisationV3 instead const Algorithm = compdesc.JsonNormalisationV2 func init() { @@ -24,6 +26,7 @@ func init() { type normalization struct{} func (m normalization) Normalize(cd *compdesc.ComponentDescriptor) ([]byte, error) { + legacy.DefaultingOfVersionIntoExtraIdentity(cd) data, err := signing.Normalize(jcs.Type, cd, CDExcludes) return data, err } diff --git a/api/ocm/compdesc/normalizations/jsonv3/norm.go b/api/ocm/compdesc/normalizations/jsonv3/norm.go new file mode 100644 index 0000000000..f8092920b0 --- /dev/null +++ b/api/ocm/compdesc/normalizations/jsonv3/norm.go @@ -0,0 +1,31 @@ +// Package jsonv3 provides a normalization which is completely based on the +// abstract (internal) version of the component descriptor and is therefore +// agnostic of the final serialization format. Signatures using this algorithm +// can be transferred among different schema versions, as long as is able to +// handle the complete information using for the normalization. +// jsonv2 is the predecessor of this version but had internal defaulting logic +// that is no longer included as part of this normalization. Thus v3 should be preferred over v2. +// Note that between v2 and v3 differences can occur mainly if the "extra identity" field is not unique, +// in which case the v2 normalization opinionated on how to differentiate these items. This no longer +// happens in v3, meaning the component descriptor is normalized as is. +package jsonv3 + +import ( + "ocm.software/ocm/api/ocm/compdesc" + "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv2" + "ocm.software/ocm/api/tech/signing" + "ocm.software/ocm/api/tech/signing/norm/jcs" +) + +const Algorithm = compdesc.JsonNormalisationV3 + +func init() { + compdesc.Normalizations.Register(Algorithm, normalization{}) +} + +type normalization struct{} + +func (m normalization) Normalize(cd *compdesc.ComponentDescriptor) ([]byte, error) { + data, err := signing.Normalize(jcs.Type, cd, jsonv2.CDExcludes) + return data, err +} diff --git a/api/ocm/compdesc/normalizations/legacy/legacy.go b/api/ocm/compdesc/normalizations/legacy/legacy.go new file mode 100644 index 0000000000..e05c982035 --- /dev/null +++ b/api/ocm/compdesc/normalizations/legacy/legacy.go @@ -0,0 +1,57 @@ +package legacy + +import ( + "fmt" + + "ocm.software/ocm/api/ocm/compdesc" + "ocm.software/ocm/api/ocm/selectors/accessors" + "ocm.software/ocm/api/utils/logging" +) + +var ( + REALM = logging.DefineSubRealm("component descriptor legacy normalization defaulting", "compdesc", "normalizations", "legacy") + Logger = logging.DynamicLogger(REALM) +) + +// DefaultingOfVersionIntoExtraIdentity normalizes the extra identity of the resources. +// It sets the version of the resource, reference or source as extra identity field if the combination of name+extra identity +// is the same for multiple items. However, the last item in the list will not be updated as it is unique wihout this. +// +// TODO: To be removed once v1 + v2 are removed. +// +// Deprecated: This is a legacy normalization and should only be used as part of JsonNormalisationV1 and JsonNormalisationV2 +// for backwards compatibility of normalization (for example used for signatures). It was needed because the original +// defaulting was made part of the normalization by accident and is now no longer included by default due to +// https://github.com/open-component-model/ocm/pull/1026 +func DefaultingOfVersionIntoExtraIdentity(cd *compdesc.ComponentDescriptor) { + resources := make([]accessors.ElementMeta, len(cd.Resources)) + for i := range cd.Resources { + resources[i] = &cd.Resources[i] + } + defaultingOfVersionIntoExtraIdentity(resources) +} + +func defaultingOfVersionIntoExtraIdentity(meta []accessors.ElementMeta) { + for i := range meta { + for j := range meta { + // don't match with itself and only match with the same name + if meta[j].GetName() != meta[i].GetName() || i == j { + continue + } + + eid := meta[i].GetExtraIdentity() + // if the extra identity is not the same, then there is not a clash + if !meta[j].GetExtraIdentity().Equals(eid) { + continue + } + + eid.Set(compdesc.SystemIdentityVersion, meta[i].GetVersion()) + meta[i].GetMeta().SetExtraIdentity(eid) + + Logger.Warn(fmt.Sprintf("resource identity duplication was normalized for backwards compatibility, "+ + "to avoid this either specify a unique extra identity per item or switch to %s", compdesc.JsonNormalisationV3), + "name", meta[i].GetName(), "index", i, "extra identity", meta[i].GetExtraIdentity()) + break + } + } +} diff --git a/api/ocm/selectors/accessors/accessors.go b/api/ocm/selectors/accessors/accessors.go index fa8b776487..9fa3c8e00a 100644 --- a/api/ocm/selectors/accessors/accessors.go +++ b/api/ocm/selectors/accessors/accessors.go @@ -27,6 +27,7 @@ type ElementMeta interface { GetMeta() ElementMeta // ElementMeta is again a Meta provider SetLabels(labels []v1.Label) + SetExtraIdentity(identity v1.Identity) } // ElementMetaProvider just provides access to element meta data diff --git a/api/ocm/tools/toi/drivers/docker/driver.go b/api/ocm/tools/toi/drivers/docker/driver.go index 75a15c25f1..c6fd47a9a1 100644 --- a/api/ocm/tools/toi/drivers/docker/driver.go +++ b/api/ocm/tools/toi/drivers/docker/driver.go @@ -469,7 +469,7 @@ func generateTar(files map[string]blobaccess.BlobAccess, uid int) (io.ReadCloser dir := path for dir != "/" { dir = unix_path.Dir(dir) - if !have[dir] { + if dir != "/" && dir != "." && !have[dir] { dirHdr := &tar.Header{ Typeflag: tar.TypeDir, Name: dir, diff --git a/cmds/ocm/commands/ocmcmds/common/options/hashoption/option.go b/cmds/ocm/commands/ocmcmds/common/options/hashoption/option.go index 4eb900f449..04a755e25e 100644 --- a/cmds/ocm/commands/ocmcmds/common/options/hashoption/option.go +++ b/cmds/ocm/commands/ocmcmds/common/options/hashoption/option.go @@ -7,6 +7,8 @@ import ( clictx "ocm.software/ocm/api/cli" "ocm.software/ocm/api/ocm/compdesc" "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv1" + "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv2" + "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv3" "ocm.software/ocm/api/ocm/extensions/attrs/signingattr" ocmsign "ocm.software/ocm/api/ocm/tools/signing" "ocm.software/ocm/api/tech/signing" @@ -34,13 +36,13 @@ type Option struct { } func (o *Option) AddFlags(fs *pflag.FlagSet) { - fs.StringVarP(&o.NormAlgorithm, "normalization", "N", jsonv1.Algorithm, "normalization algorithm") + fs.StringVarP(&o.NormAlgorithm, "normalization", "N", jsonv3.Algorithm, "normalization algorithm") fs.StringVarP(&o.hashAlgorithm, "hash", "H", sha256.Algorithm, "hash algorithm") } func (o *Option) Configure(ctx clictx.Context) error { if o.NormAlgorithm == "" { - o.NormAlgorithm = jsonv1.Algorithm + o.NormAlgorithm = jsonv3.Algorithm } if o.hashAlgorithm == "" { o.hashAlgorithm = sha256.Algorithm @@ -59,7 +61,18 @@ func (o *Option) Configure(ctx clictx.Context) error { func (o *Option) Usage() string { s := ` The following normalization modes are supported with option --normalization: -` + listformat.FormatList(jsonv1.Algorithm, compdesc.Normalizations.Names()...) +` + listformat.FormatList(jsonv3.Algorithm, compdesc.Normalizations.Names()...) + + s += ` + +Note that the normalization algorithm is important to be equivalent when used for signing and verification, otherwise +the verification can fail. Please always migrate to the latest normalization algorithm whenever possible. +New signature algorithms can be used as soon as they are available in the component version after signing it. + +The algorithms ` + jsonv1.Algorithm + ` and ` + jsonv2.Algorithm + ` are deprecated and should not be used anymore. +Please switch to ` + jsonv3.Algorithm + ` as soon as possible. + +` s += ` diff --git a/cmds/ocm/commands/ocmcmds/common/options/signoption/option.go b/cmds/ocm/commands/ocmcmds/common/options/signoption/option.go index 14723d097a..a0ac7fdb22 100644 --- a/cmds/ocm/commands/ocmcmds/common/options/signoption/option.go +++ b/cmds/ocm/commands/ocmcmds/common/options/signoption/option.go @@ -9,7 +9,7 @@ import ( clictx "ocm.software/ocm/api/cli" "ocm.software/ocm/api/ocm/compdesc" - "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv1" + "ocm.software/ocm/api/ocm/compdesc/normalizations/jsonv3" "ocm.software/ocm/api/ocm/extensions/attrs/signingattr" ocmsign "ocm.software/ocm/api/ocm/tools/signing" "ocm.software/ocm/api/tech/signing" @@ -151,7 +151,7 @@ The following signing types are supported with option --algorithm: s += ` The following normalization modes are supported with option --normalization: -` + listformat.FormatList(jsonv1.Algorithm, compdesc.Normalizations.Names()...) +` + listformat.FormatList(jsonv3.Algorithm, compdesc.Normalizations.Names()...) s += ` diff --git a/cmds/ocm/commands/ocmcmds/components/hash/cmd_test.go b/cmds/ocm/commands/ocmcmds/components/hash/cmd_test.go index f79e2c1c05..6f20c09d0c 100644 --- a/cmds/ocm/commands/ocmcmds/components/hash/cmd_test.go +++ b/cmds/ocm/commands/ocmcmds/components/hash/cmd_test.go @@ -9,6 +9,7 @@ import ( . "github.com/mandelsoft/goutils/testutils" . "github.com/onsi/ginkgo/v2" . "github.com/onsi/gomega" + . "ocm.software/ocm/cmds/ocm/testhelper" "ocm.software/ocm/api/ocm/compdesc" @@ -38,41 +39,47 @@ var _ = Describe("Test Environment", func() { env.Cleanup() }) - It("hash component archive", func() { - env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { - env.Provider(PROVIDER) - }) - - buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-o", "wide")).To(Succeed()) - Expect(buf.String()).To(StringEqualTrimmedWithContext( - ` -COMPONENT VERSION HASH NORMALIZED FORM -test.de/x v1 37f7f500d87f4b0a8765649f7c047db382e272b73e042805131df57279991b2b [{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] -`)) - }) - - It("normalize component archive v1", func() { - env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { - env.Provider(PROVIDER) - }) + DescribeTable("should hash the component archive with specified parameters", + func(normalizationMethod string, expectedOutput string) { + env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { + env.Provider(PROVIDER) + }) - buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-O", "-", "-o", "norm")).To(Succeed()) - Expect(buf.String()).To(Equal(`[{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] -`)) - }) + buf := bytes.NewBuffer(nil) + cmdArgs := []string{"hash", "components", ARCH, "-o", "wide", "--normalization", normalizationMethod} + Expect(env.CatchOutput(buf).Execute(cmdArgs...)).To(Succeed()) + Expect(buf.String()).To(StringEqualTrimmedWithContext(expectedOutput)) + }, - It("normalize component archive v2", func() { + Entry("v1", compdesc.JsonNormalisationV1, + `COMPONENT VERSION HASH NORMALIZED FORM +test.de/x v1 37f7f500d87f4b0a8765649f7c047db382e272b73e042805131df57279991b2b [{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] +`), + Entry("v2", compdesc.JsonNormalisationV2, + `COMPONENT VERSION HASH NORMALIZED FORM +test.de/x v1 33aeb8c46ea4bacbf0bc3ac42c186c7f5e313584601a93bf861c016d73c9e4f1 {"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}} +`), + Entry("v3", compdesc.JsonNormalisationV3, + `COMPONENT VERSION HASH NORMALIZED FORM +test.de/x v1 33aeb8c46ea4bacbf0bc3ac42c186c7f5e313584601a93bf861c016d73c9e4f1 {"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}} +`), + ) + + DescribeTable("normalize component archive", func(normalizationMethod string, expectedOutput string) { env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { env.Provider(PROVIDER) }) buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-N", "jsonNormalisation/v2", "-o", "norm")).To(Succeed()) - Expect(buf.String()).To(StringEqualTrimmedWithContext(`{"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}} + Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-O", "-", "-o", "norm", "-N", normalizationMethod)).To(Succeed()) + Expect(buf.String()).To(Equal(expectedOutput)) + }, + Entry("v1", compdesc.JsonNormalisationV1, `[{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] +`), + Entry("v2", compdesc.JsonNormalisationV2, `{"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}} +`), + Entry("v3", compdesc.JsonNormalisationV3, `{"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}} `)) - }) It("check hash", func() { env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { @@ -85,16 +92,16 @@ test.de/x v1 37f7f500d87f4b0a8765649f7c047db382e272b73e042805131df57279991b --- component: test.de/x context: [] -hash: 37f7f500d87f4b0a8765649f7c047db382e272b73e042805131df57279991b2b -normalized: '[{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}]' +hash: 33aeb8c46ea4bacbf0bc3ac42c186c7f5e313584601a93bf861c016d73c9e4f1 +normalized: '{"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}}' version: v1 `)) - h := sha256.Sum256([]byte(`[{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}]`)) - Expect(hex.EncodeToString(h[:])).To(Equal("37f7f500d87f4b0a8765649f7c047db382e272b73e042805131df57279991b2b")) + h := sha256.Sum256([]byte(`{"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[],"sources":[],"version":"v1"}}`)) + Expect(hex.EncodeToString(h[:])).To(Equal("33aeb8c46ea4bacbf0bc3ac42c186c7f5e313584601a93bf861c016d73c9e4f1")) }) - It("hash component archive with resources", func() { + DescribeTable("hash component archive with resources", func(normalizationMethod string, expectedOutput string) { env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { env.Provider(PROVIDER) env.Resource("test", VERSION, resourcetypes.PLAIN_TEXT, metav1.LocalRelation, func() { @@ -103,34 +110,27 @@ version: v1 }) buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-o", "wide")).To(Succeed()) - Expect(buf.String()).To(StringEqualTrimmedWithContext( - ` + Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "-o", "wide", "-N", normalizationMethod)).To(Succeed()) + Expect(buf.String()).To(StringEqualTrimmedWithContext(expectedOutput)) + }, + Entry("v1", compdesc.JsonNormalisationV1, ` COMPONENT : test.de/x VERSION : v1 HASH : 9d8fc24cf27d1092f58098286d9f63c6824c2daf739c19789f64c062d1f30cc5 NORMALIZED FORM: [{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[[{"digest":[{"hashAlgorithm":"SHA-256"},{"normalisationAlgorithm":"genericBlobDigest/v1"},{"value":"810ff2fb242a5dee4220f2cb0e6a519891fb67f2f828a6cab4ef8894633b1f50"}]},{"extraIdentity":null},{"name":"test"},{"relation":"local"},{"type":"plainText"},{"version":"v1"}]]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] ----`)) - }) - - It("hash component archive with resources", func() { - env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { - env.Provider(PROVIDER) - env.Resource("test", VERSION, resourcetypes.PLAIN_TEXT, metav1.LocalRelation, func() { - env.BlobStringData(mime.MIME_TEXT, "testdata") - }) - }) - - buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", ARCH, "--actual", "-o", "wide")).To(Succeed()) - Expect(buf.String()).To(StringEqualTrimmedWithContext( - ` +---`), + Entry("v2", compdesc.JsonNormalisationV2, ` COMPONENT : test.de/x VERSION : v1 -HASH : 9d8fc24cf27d1092f58098286d9f63c6824c2daf739c19789f64c062d1f30cc5 -NORMALIZED FORM: [{"component":[{"componentReferences":[]},{"name":"test.de/x"},{"provider":"mandelsoft"},{"resources":[[{"digest":[{"hashAlgorithm":"SHA-256"},{"normalisationAlgorithm":"genericBlobDigest/v1"},{"value":"810ff2fb242a5dee4220f2cb0e6a519891fb67f2f828a6cab4ef8894633b1f50"}]},{"extraIdentity":null},{"name":"test"},{"relation":"local"},{"type":"plainText"},{"version":"v1"}]]},{"version":"v1"}]},{"meta":[{"schemaVersion":"v2"}]}] +HASH : 6e8e9eb0af1c4c0b9dcc4161168b3f0ad913bc85e4234688dd6d4b283fe4b956 +NORMALIZED FORM: {"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"genericBlobDigest/v1","value":"810ff2fb242a5dee4220f2cb0e6a519891fb67f2f828a6cab4ef8894633b1f50"},"name":"test","relation":"local","type":"plainText","version":"v1"}],"sources":[],"version":"v1"}} +---`), + Entry("v3", compdesc.JsonNormalisationV3, ` +COMPONENT : test.de/x +VERSION : v1 +HASH : 6e8e9eb0af1c4c0b9dcc4161168b3f0ad913bc85e4234688dd6d4b283fe4b956 +NORMALIZED FORM: {"component":{"componentReferences":[],"name":"test.de/x","provider":{"name":"mandelsoft"},"resources":[{"digest":{"hashAlgorithm":"SHA-256","normalisationAlgorithm":"genericBlobDigest/v1","value":"810ff2fb242a5dee4220f2cb0e6a519891fb67f2f828a6cab4ef8894633b1f50"},"name":"test","relation":"local","type":"plainText","version":"v1"}],"sources":[],"version":"v1"}} ---`)) - }) It("hash component archive with v2", func() { env.ComponentArchive(ARCH, accessio.FormatDirectory, COMP, VERSION, func() { @@ -151,7 +151,7 @@ NORMALIZED FORM: {"component":{"componentReferences":[],"name":"test.de/x","prov ---`)) }) - It("hash component recursively", func() { + It("hash partial component archive recursively", func() { env.OCMCommonTransport(ARCH, accessio.FormatDirectory, func() { env.ComponentVersion(COMP2, VERSION, func() { env.Provider(PROVIDER) @@ -166,32 +166,12 @@ NORMALIZED FORM: {"component":{"componentReferences":[],"name":"test.de/x","prov Expect(env.CatchOutput(buf).Execute("hash", "components", "-r", ARCH+"//test.de/x:v1")).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` REFERENCEPATH COMPONENT VERSION HASH IDENTITY - test.de/x v1 b74cee6c6b8215f470efd0e3c49618bb98610fc80de36a2e121d0550650b9cdc -test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a "name"="ref" -`)) - }) - - It("hash component recursively", func() { - env.OCMCommonTransport(ARCH, accessio.FormatDirectory, func() { - env.ComponentVersion(COMP2, VERSION, func() { - env.Provider(PROVIDER) - }) - env.ComponentVersion(COMP, VERSION, func() { - env.Provider(PROVIDER) - env.Reference("ref", COMP2, VERSION) - }) - }) - - buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("hash", "components", "-r", "--repo", ARCH, "test.de/x:v1")).To(Succeed()) - Expect(buf.String()).To(StringEqualTrimmedWithContext(` -REFERENCEPATH COMPONENT VERSION HASH IDENTITY - test.de/x v1 b74cee6c6b8215f470efd0e3c49618bb98610fc80de36a2e121d0550650b9cdc -test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a "name"="ref" + test.de/x v1 4ca827281d94cdfee77e7ab2d89164f5e0d38890d12f035c2cbe19d72851fb17 +test.de/x:v1 test.de/y v1 bd420aea257660f444b39165cee42b905e48165021781e2468dcf8e1cbc0151d "name"="ref" `)) }) - It("hash components recursively", func() { + It("hash component archive recursively", func() { env.OCMCommonTransport(ARCH, accessio.FormatDirectory, func() { env.ComponentVersion(COMP2, VERSION, func() { env.Provider(PROVIDER) @@ -206,9 +186,9 @@ test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd7218 Expect(env.CatchOutput(buf).Execute("hash", "components", "-r", ARCH)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` REFERENCEPATH COMPONENT VERSION HASH IDENTITY - test.de/x v1 b74cee6c6b8215f470efd0e3c49618bb98610fc80de36a2e121d0550650b9cdc -test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a "name"="ref" - test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a + test.de/x v1 4ca827281d94cdfee77e7ab2d89164f5e0d38890d12f035c2cbe19d72851fb17 +test.de/x:v1 test.de/y v1 bd420aea257660f444b39165cee42b905e48165021781e2468dcf8e1cbc0151d "name"="ref" + test.de/y v1 bd420aea257660f444b39165cee42b905e48165021781e2468dcf8e1cbc0151d `)) }) @@ -227,8 +207,8 @@ test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd7218 Expect(env.CatchOutput(buf).Execute("hash", "components", "-r", "--repo", ARCH, "-U", "test.de/x:v1")).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` REFERENCEPATH COMPONENT VERSION HASH IDENTITY - test.de/x v1 b74cee6c6b8215f470efd0e3c49618bb98610fc80de36a2e121d0550650b9cdc -test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a "name"="ref" + test.de/x v1 4ca827281d94cdfee77e7ab2d89164f5e0d38890d12f035c2cbe19d72851fb17 +test.de/x:v1 test.de/y v1 bd420aea257660f444b39165cee42b905e48165021781e2468dcf8e1cbc0151d "name"="ref" `)) repo := Must(ctf.Open(env, ctf.ACC_READONLY, ARCH, 0, env)) @@ -247,6 +227,6 @@ test.de/x:v1 test.de/y v1 e60c791a20091abcf8d35742a134b3a99ce811d874fd7218 ref := Must(cv.GetReferenceByIndex(0)) d := ref.GetDigest() Expect(d).NotTo(BeNil()) - Expect(d.Value).To(Equal("e60c791a20091abcf8d35742a134b3a99ce811d874fd721870b28ea90ef5ad2a")) + Expect(d.Value).To(Equal("bd420aea257660f444b39165cee42b905e48165021781e2468dcf8e1cbc0151d")) }) }) diff --git a/cmds/ocm/commands/ocmcmds/components/sign/cmd_test.go b/cmds/ocm/commands/ocmcmds/components/sign/cmd_test.go index 0537457516..7991b3f843 100644 --- a/cmds/ocm/commands/ocmcmds/components/sign/cmd_test.go +++ b/cmds/ocm/commands/ocmcmds/components/sign/cmd_test.go @@ -7,7 +7,9 @@ import ( . "github.com/mandelsoft/goutils/testutils" . "github.com/onsi/ginkgo/v2" . "github.com/onsi/gomega" + . "ocm.software/ocm/api/oci/testhelper" + "ocm.software/ocm/api/ocm/compdesc" . "ocm.software/ocm/api/ocm/testhelper" . "ocm.software/ocm/cmds/ocm/testhelper" @@ -52,19 +54,50 @@ const ( ) const ( - D_COMPONENTA = "01de99400030e8336020059a435cea4e7fe8f21aad4faf619da882134b85569d" - D_COMPONENTB = "5f416ec59629d6af91287e2ba13c6360339b6a0acf624af2abd2a810ce4aefce" + D_COMPONENTA_V1 = "01de99400030e8336020059a435cea4e7fe8f21aad4faf619da882134b85569d" + D_COMPONENTB_V1 = "5f416ec59629d6af91287e2ba13c6360339b6a0acf624af2abd2a810ce4aefce" ) const VERIFIED_FILE = "verified.yaml" -var substitutions = Substitutions{ - "test": D_COMPONENTA, - "r0": D_TESTDATA, - "r1": DS_OCIMANIFEST1.Value, - "r2": DS_OCIMANIFEST2.Value, - "ref": D_COMPONENTB, - "rb0": D_OTHERDATA, +var substitutionsV1 = Substitutions{ + "test": D_COMPONENTA_V1, + "r0": D_TESTDATA, + "r1": DS_OCIMANIFEST1.Value, + "r2": DS_OCIMANIFEST2.Value, + "ref": D_COMPONENTB_V1, + "rb0": D_OTHERDATA, + "normAlgo": compdesc.JsonNormalisationV1, +} + +const ( + D_COMPONENTA_V2 = "10ac0b3a850e1f1becf56d5d45e9742fa0a91103d25ba93cc3a509f68797e90f" + D_COMPONENTB_V2 = "1ae74420ef29436ad75133d81bceb59fa8ef1e2ce083a45b5f4baaec641a4266" +) + +var substitutionsV2 = Substitutions{ + "test": D_COMPONENTA_V2, + "r0": D_TESTDATA, + "r1": DS_OCIMANIFEST1.Value, + "r2": DS_OCIMANIFEST2.Value, + "ref": D_COMPONENTB_V2, + "rb0": D_OTHERDATA, + "normAlgo": compdesc.JsonNormalisationV2, +} + +const ( + D_COMPONENTA_V3 = D_COMPONENTA_V2 + D_COMPONENTB_V3 = "766f26b09237f9647714e85fac914f115d0b4c3277b01ec00cfeb3b50a68cde9" +) + +var substitutionsV3 = Substitutions{ + "test": D_COMPONENTA_V3, + "r0": D_TESTDATA, + "r1": DS_OCIMANIFEST1.Value, + "r2": DS_OCIMANIFEST2.Value, + "ref": D_COMPONENTB_V3, + "rb0": D_OTHERDATA, + "normAlgo": compdesc.JsonNormalisationV3, } var _ = Describe("access method", func() { @@ -116,7 +149,7 @@ var _ = Describe("access method", func() { prepareEnv(env, ARCH, "") buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTA+":"+VERSION)).To(Succeed()) + Expect(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTA+":"+VERSION, "--normalization", compdesc.JsonNormalisationV1)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/test:v1"[github.com/mandelsoft/test:v1]... @@ -124,7 +157,7 @@ applying to version "github.com/mandelsoft/test:v1"[github.com/mandelsoft/test:v resource 1: "name"="value": digest SHA-256:${r1}[ociArtifactDigest/v1] resource 2: "name"="ref": digest SHA-256:${r2}[ociArtifactDigest/v1] successfully signed github.com/mandelsoft/test:v1 (digest SHA-256:${test})`, - substitutions), + substitutionsV1), ) session := datacontext.NewSession() @@ -136,14 +169,14 @@ successfully signed github.com/mandelsoft/test:v1 (digest SHA-256:${test})`, cv, err := src.LookupComponentVersion(COMPONENTA, VERSION) Expect(err).To(Succeed()) session.AddCloser(cv) - Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTA)) + Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTA_V1)) }) It("signs transport archive", func() { prepareEnv(env, ARCH, ARCH) buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION)).To(Succeed()) + Expect(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION, "--normalization", compdesc.JsonNormalisationV1)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1]... @@ -155,7 +188,7 @@ applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1] reference 0: github.com/mandelsoft/test:v1: digest SHA-256:${test}[jsonNormalisation/v1] resource 0: "name"="otherdata": digest SHA-256:${rb0}[genericBlobDigest/v1] successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) -`, substitutions)) +`, substitutionsV1)) session := datacontext.NewSession() defer session.Close() @@ -166,14 +199,14 @@ successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) cv, err := src.LookupComponentVersion(COMPONENTB, VERSION) Expect(err).To(Succeed()) session.AddCloser(cv) - Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTB)) + Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTB_V1)) }) It("signs transport archive with --lookup option", func() { prepareEnv(env, ARCH2, ARCH) buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("sign", "components", "--lookup", ARCH2, "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION)).To(Succeed()) + Expect(env.CatchOutput(buf).Execute("sign", "components", "--lookup", ARCH2, "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION, "--normalization", compdesc.JsonNormalisationV1)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1]... @@ -185,7 +218,7 @@ applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1] reference 0: github.com/mandelsoft/test:v1: digest SHA-256:${test}[jsonNormalisation/v1] resource 0: "name"="otherdata": digest SHA-256:${rb0}[genericBlobDigest/v1] successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) -`, substitutions)) +`, substitutionsV1)) session := datacontext.NewSession() defer session.Close() @@ -196,7 +229,7 @@ successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) cv, err := src.LookupComponentVersion(COMPONENTB, VERSION) Expect(err).To(Succeed()) session.AddCloser(cv) - Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTB)) + Expect(cv.GetDescriptor().Signatures[0].Digest.Value).To(Equal(D_COMPONENTB_V1)) }) }) @@ -281,7 +314,7 @@ Error: signing: github.com/mandelsoft/ref:v1: failed resolving component referen It("signs comp arch with lookup", func() { buf := bytes.NewBuffer(nil) - MustBeSuccessful(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--lookup", ARCH, "--repo", COMPARCH)) + MustBeSuccessful(env.CatchOutput(buf).Execute("sign", "components", "-s", SIGNATURE, "-K", PRIVKEY, "--lookup", ARCH, "--repo", COMPARCH, "--normalization", compdesc.JsonNormalisationV1)) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1]... no digest found for "github.com/mandelsoft/test:v1" @@ -330,7 +363,7 @@ created rsa key pair key.priv[key.cert] // sigh component with certificate buf.Reset() - Expect(env.CatchOutput(buf).Execute("sign", "component", ARCH, "-K", "key.priv", "-k", "key.cert", "--ca-cert", "root.cert", "-s", "mandelsoft", "-I", "CN=mandelsoft")).To(Succeed()) + Expect(env.CatchOutput(buf).Execute("sign", "component", ARCH, "-K", "key.priv", "-k", "key.cert", "--ca-cert", "root.cert", "-s", "mandelsoft", "-I", "CN=mandelsoft", "--normalization", compdesc.JsonNormalisationV1)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/test:v1"[github.com/mandelsoft/test:v1]... successfully signed github.com/mandelsoft/test:v1 (digest SHA-256:5ed8bb27309c3c2fff43f3b0f3ebb56a5737ad6db4bc8ace73c5455cb86faf54) @@ -371,11 +404,11 @@ successfully verified github.com/mandelsoft/test:v1 (digest SHA-256:5ed8bb27309c }) }) - It("signs transport archive", func() { + DescribeTable("signs transport archive", func(substitutions Substitutions, normAlgo string) { prepareEnv(env, ARCH, ARCH) buf := bytes.NewBuffer(nil) - Expect(env.CatchOutput(buf).Execute("sign", "components", "--verified", VERIFIED_FILE, "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION)).To(Succeed()) + Expect(env.CatchOutput(buf).Execute("sign", "components", "--verified", VERIFIED_FILE, "-s", SIGNATURE, "-K", PRIVKEY, "--repo", ARCH, COMPONENTB+":"+VERSION, "--normalization", normAlgo)).To(Succeed()) Expect(buf.String()).To(StringEqualTrimmedWithContext(` applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1]... @@ -384,7 +417,7 @@ applying to version "github.com/mandelsoft/ref:v1"[github.com/mandelsoft/ref:v1] resource 0: "name"="testdata": digest SHA-256:${r0}[genericBlobDigest/v1] resource 1: "name"="value": digest SHA-256:${r1}[ociArtifactDigest/v1] resource 2: "name"="ref": digest SHA-256:${r2}[ociArtifactDigest/v1] - reference 0: github.com/mandelsoft/test:v1: digest SHA-256:${test}[jsonNormalisation/v1] + reference 0: github.com/mandelsoft/test:v1: digest SHA-256:${test}[${normAlgo}] resource 0: "name"="otherdata": digest SHA-256:${rb0}[genericBlobDigest/v1] successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) `, substitutions)) @@ -395,7 +428,11 @@ successfully signed github.com/mandelsoft/ref:v1 (digest SHA-256:${ref}) CheckStore(store, common.NewNameVersion(COMPONENTA, VERSION)) CheckStore(store, common.NewNameVersion(COMPONENTB, VERSION)) - }) + }, + Entry("v1", substitutionsV1, compdesc.JsonNormalisationV1), + Entry("v2", substitutionsV2, compdesc.JsonNormalisationV2), + Entry("v3", substitutionsV3, compdesc.JsonNormalisationV3), + ) }) }) diff --git a/docs/reference/ocm_hash_componentversions.md b/docs/reference/ocm_hash_componentversions.md index 1e49ba07f4..4c5e46bdca 100644 --- a/docs/reference/ocm_hash_componentversions.md +++ b/docs/reference/ocm_hash_componentversions.md @@ -21,7 +21,7 @@ componentversions, componentversion, cv, components, component, comps, comp, c -h, --help help for componentversions --latest restrict component versions to latest --lookup stringArray repository name or spec for closure lookup fallback - -N, --normalization string normalization algorithm (default "jsonNormalisation/v1") + -N, --normalization string normalization algorithm (default "jsonNormalisation/v3") -O, --outfile string Output file for normalized component descriptor (default "-") -o, --output string output mode (JSON, json, norm, wide, yaml) -r, --recursive follow component reference nesting @@ -63,8 +63,18 @@ references. The following normalization modes are supported with option --normalization: - - jsonNormalisation/v1 (default) + - jsonNormalisation/v1 - jsonNormalisation/v2 + - jsonNormalisation/v3 (default) + + +Note that the normalization algorithm is important to be equivalent when used for signing and verification, otherwise +the verification can fail. Please always migrate to the latest normalization algorithm whenever possible. +New signature algorithms can be used as soon as they are available in the component version after signing it. + +The algorithms jsonNormalisation/v1 and jsonNormalisation/v2 are deprecated and should not be used anymore. +Please switch to jsonNormalisation/v3 as soon as possible. + The following hash modes are supported with option --hash: diff --git a/docs/reference/ocm_logging.md b/docs/reference/ocm_logging.md index c73dc6cc3e..282772ff96 100644 --- a/docs/reference/ocm_logging.md +++ b/docs/reference/ocm_logging.md @@ -21,6 +21,7 @@ The following *realms* are used by the command line tool: - ocm/accessmethod/wget: access method for wget - ocm/blobaccess/wget: blob access for wget - ocm/compdesc: component descriptor handling + - ocm/compdesc/normalizations/legacy: component descriptor legacy normalization defaulting - ocm/config: configuration management - ocm/context: context lifecycle - ocm/credentials: Credentials diff --git a/docs/reference/ocm_sign_componentversions.md b/docs/reference/ocm_sign_componentversions.md index b1d3f582bf..fa9a0d7d17 100644 --- a/docs/reference/ocm_sign_componentversions.md +++ b/docs/reference/ocm_sign_componentversions.md @@ -25,7 +25,7 @@ componentversions, componentversion, cv, components, component, comps, comp, c --keyless use keyless signing --latest restrict component versions to latest --lookup stringArray repository name or spec for closure lookup fallback - -N, --normalization string normalization algorithm (default "jsonNormalisation/v1") + -N, --normalization string normalization algorithm (default "jsonNormalisation/v3") -K, --private-key stringArray private key setting -k, --public-key stringArray public key setting -R, --recursive recursively sign component versions @@ -124,8 +124,9 @@ The following signing types are supported with option --algorithm: The following normalization modes are supported with option --normalization: - - jsonNormalisation/v1 (default) + - jsonNormalisation/v1 - jsonNormalisation/v2 + - jsonNormalisation/v3 (default) The following hash modes are supported with option --hash: