From 1adde98228cd2a2e6f71d38a1be97907c11d9a14 Mon Sep 17 00:00:00 2001 From: Joe Lanford Date: Thu, 12 Sep 2024 05:23:11 -0400 Subject: [PATCH] template/*: allow passing custom bundle renderer (#1423) --- alpha/action/list_test.go | 6 ++-- alpha/action/render.go | 8 ++--- alpha/template/basic/basic.go | 16 ++-------- alpha/template/semver/semver.go | 45 +++++++++++++--------------- alpha/template/semver/semver_test.go | 4 +-- alpha/template/semver/types.go | 9 +++--- cmd/opm/alpha/template/basic.go | 19 +++++++++--- cmd/opm/alpha/template/semver.go | 24 +++++++++++---- 8 files changed, 69 insertions(+), 62 deletions(-) diff --git a/alpha/action/list_test.go b/alpha/action/list_test.go index 41dbc99cc..090439d44 100644 --- a/alpha/action/list_test.go +++ b/alpha/action/list_test.go @@ -28,7 +28,7 @@ foo Foo Operator beta { name: "Error/UnknownIndex", list: ListPackages{IndexReference: "unknown-index"}, - expectedErr: `render reference "unknown-index": repository name must be canonical`, + expectedErr: `render reference "unknown-index": failed to pull image "unknown-index": repository name must be canonical`, }, } for _, s := range specs { @@ -79,7 +79,7 @@ foo stable foo.v0.2.0 { name: "Error/UnknownIndex", list: ListChannels{IndexReference: "unknown-index"}, - expectedErr: `render reference "unknown-index": repository name must be canonical`, + expectedErr: `render reference "unknown-index": failed to pull image "unknown-index": repository name must be canonical`, }, { name: "Error/UnknownPackage", @@ -138,7 +138,7 @@ foo stable foo.v0.2.0 foo.v0.1.0 foo.v0.1.1,foo.v0.1.2 <0.2.0 tes { name: "Error/UnknownIndex", list: ListBundles{IndexReference: "unknown-index"}, - expectedErr: `render reference "unknown-index": repository name must be canonical`, + expectedErr: `render reference "unknown-index": failed to pull image "unknown-index": repository name must be canonical`, }, { name: "Error/UnknownPackage", diff --git a/alpha/action/render.go b/alpha/action/render.go index 798ffb5e5..07631b7c4 100644 --- a/alpha/action/render.go +++ b/alpha/action/render.go @@ -162,19 +162,19 @@ func (r Render) renderReference(ctx context.Context, ref string) (*declcfg.Decla func (r Render) imageToDeclcfg(ctx context.Context, imageRef string) (*declcfg.DeclarativeConfig, error) { ref := image.SimpleReference(imageRef) if err := r.Registry.Pull(ctx, ref); err != nil { - return nil, err + return nil, fmt.Errorf("failed to pull image %q: %v", ref, err) } labels, err := r.Registry.Labels(ctx, ref) if err != nil { - return nil, err + return nil, fmt.Errorf("failed to get labels for image %q: %v", ref, err) } tmpDir, err := os.MkdirTemp("", "render-unpack-") if err != nil { - return nil, err + return nil, fmt.Errorf("create tempdir: %v", err) } defer os.RemoveAll(tmpDir) if err := r.Registry.Unpack(ctx, ref, tmpDir); err != nil { - return nil, err + return nil, fmt.Errorf("failed to unpack image %q: %v", ref, err) } var cfg *declcfg.DeclarativeConfig diff --git a/alpha/template/basic/basic.go b/alpha/template/basic/basic.go index 0456dc151..a34d7541d 100644 --- a/alpha/template/basic/basic.go +++ b/alpha/template/basic/basic.go @@ -8,17 +8,13 @@ import ( "k8s.io/apimachinery/pkg/util/yaml" - "github.com/operator-framework/operator-registry/alpha/action" - "github.com/operator-framework/operator-registry/alpha/action/migrations" "github.com/operator-framework/operator-registry/alpha/declcfg" - "github.com/operator-framework/operator-registry/pkg/image" ) const schema string = "olm.template.basic" type Template struct { - Registry image.Registry - Migrations *migrations.Migrations + RenderBundle func(context.Context, string) (*declcfg.DeclarativeConfig, error) } type BasicTemplate struct { @@ -57,19 +53,11 @@ func (t Template) Render(ctx context.Context, reader io.Reader) (*declcfg.Declar } outb := cfg.Bundles[:0] - // populate registry, incl any flags from CLI, and enforce only rendering bundle images - r := action.Render{ - Registry: t.Registry, - AllowedRefMask: action.RefBundleImage, - Migrations: t.Migrations, - } - for _, b := range cfg.Bundles { if !isBundleTemplate(&b) { return nil, fmt.Errorf("unexpected fields present in basic template bundle") } - r.Refs = []string{b.Image} - contributor, err := r.Run(ctx) + contributor, err := t.RenderBundle(ctx, b.Image) if err != nil { return nil, err } diff --git a/alpha/template/semver/semver.go b/alpha/template/semver/semver.go index dfd584a1c..d44e1c9d9 100644 --- a/alpha/template/semver/semver.go +++ b/alpha/template/semver/semver.go @@ -10,7 +10,6 @@ import ( "k8s.io/apimachinery/pkg/util/errors" "sigs.k8s.io/yaml" - "github.com/operator-framework/operator-registry/alpha/action" "github.com/operator-framework/operator-registry/alpha/declcfg" "github.com/operator-framework/operator-registry/alpha/property" ) @@ -25,22 +24,16 @@ func (t Template) Render(ctx context.Context) (*declcfg.DeclarativeConfig, error var cfgs []declcfg.DeclarativeConfig - bundleDict := make(map[string]struct{}) - buildBundleList(&sv.Candidate.Bundles, &bundleDict) - buildBundleList(&sv.Fast.Bundles, &bundleDict) - buildBundleList(&sv.Stable.Bundles, &bundleDict) - + bundleDict := buildBundleList(*sv) for b := range bundleDict { - r := action.Render{ - AllowedRefMask: action.RefBundleImage, - Refs: []string{b}, - Registry: t.Registry, - Migrations: t.Migrations, - } - c, err := r.Run(ctx) + c, err := t.RenderBundle(ctx, b) if err != nil { return nil, err } + if len(c.Bundles) != 1 { + return nil, fmt.Errorf("bundle reference %q resulted in %d bundles, expected 1", b, len(c.Bundles)) + } + bundleDict[b] = c.Bundles[0].Image cfgs = append(cfgs, *c) } out = *combineConfigs(cfgs) @@ -49,7 +42,7 @@ func (t Template) Render(ctx context.Context) (*declcfg.DeclarativeConfig, error return nil, fmt.Errorf("render: no bundles specified or no bundles could be rendered") } - channelBundleVersions, err := sv.getVersionsFromStandardChannels(&out) + channelBundleVersions, err := sv.getVersionsFromStandardChannels(&out, bundleDict) if err != nil { return nil, fmt.Errorf("render: unable to post-process bundle info: %v", err) } @@ -61,12 +54,16 @@ func (t Template) Render(ctx context.Context) (*declcfg.DeclarativeConfig, error return &out, nil } -func buildBundleList(bundles *[]semverTemplateBundleEntry, dict *map[string]struct{}) { - for _, b := range *bundles { - if _, ok := (*dict)[b.Image]; !ok { - (*dict)[b.Image] = struct{}{} +func buildBundleList(t semverTemplate) map[string]string { + dict := make(map[string]string) + for _, bl := range []semverTemplateChannelBundles{t.Candidate, t.Fast, t.Stable} { + for _, b := range bl.Bundles { + if _, ok := dict[b.Image]; !ok { + dict[b.Image] = b.Image + } } } + return dict } func readFile(reader io.Reader) (*semverTemplate, error) { @@ -114,10 +111,10 @@ func readFile(reader io.Reader) (*semverTemplate, error) { return &sv, nil } -func (sv *semverTemplate) getVersionsFromStandardChannels(cfg *declcfg.DeclarativeConfig) (*bundleVersions, error) { +func (sv *semverTemplate) getVersionsFromStandardChannels(cfg *declcfg.DeclarativeConfig, bundleDict map[string]string) (*bundleVersions, error) { versions := bundleVersions{} - bdm, err := sv.getVersionsFromChannel(sv.Candidate.Bundles, cfg) + bdm, err := sv.getVersionsFromChannel(sv.Candidate.Bundles, bundleDict, cfg) if err != nil { return nil, err } @@ -126,7 +123,7 @@ func (sv *semverTemplate) getVersionsFromStandardChannels(cfg *declcfg.Declarati } versions[candidateChannelArchetype] = bdm - bdm, err = sv.getVersionsFromChannel(sv.Fast.Bundles, cfg) + bdm, err = sv.getVersionsFromChannel(sv.Fast.Bundles, bundleDict, cfg) if err != nil { return nil, err } @@ -135,7 +132,7 @@ func (sv *semverTemplate) getVersionsFromStandardChannels(cfg *declcfg.Declarati } versions[fastChannelArchetype] = bdm - bdm, err = sv.getVersionsFromChannel(sv.Stable.Bundles, cfg) + bdm, err = sv.getVersionsFromChannel(sv.Stable.Bundles, bundleDict, cfg) if err != nil { return nil, err } @@ -147,7 +144,7 @@ func (sv *semverTemplate) getVersionsFromStandardChannels(cfg *declcfg.Declarati return &versions, nil } -func (sv *semverTemplate) getVersionsFromChannel(semverBundles []semverTemplateBundleEntry, cfg *declcfg.DeclarativeConfig) (map[string]semver.Version, error) { +func (sv *semverTemplate) getVersionsFromChannel(semverBundles []semverTemplateBundleEntry, bundleDict map[string]string, cfg *declcfg.DeclarativeConfig) (map[string]semver.Version, error) { entries := make(map[string]semver.Version) // we iterate over the channel bundles from the template, to: @@ -158,7 +155,7 @@ func (sv *semverTemplate) getVersionsFromChannel(semverBundles []semverTemplateB // test if the bundle specified in the template is present in the successfully-rendered bundles index := 0 for index < len(cfg.Bundles) { - if cfg.Bundles[index].Image == semverBundle.Image { + if cfg.Bundles[index].Image == bundleDict[semverBundle.Image] { break } index++ diff --git a/alpha/template/semver/semver_test.go b/alpha/template/semver/semver_test.go index 148290679..f1bc07e8a 100644 --- a/alpha/template/semver/semver_test.go +++ b/alpha/template/semver/semver_test.go @@ -538,7 +538,7 @@ func TestGetVersionsFromStandardChannel(t *testing.T) { for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { iosv := tt.sv - versions, err := iosv.getVersionsFromStandardChannels(&tt.dc) + versions, err := iosv.getVersionsFromStandardChannels(&tt.dc, buildBundleList(tt.sv)) require.NoError(t, err) require.EqualValues(t, tt.outVersions, *versions) require.EqualValues(t, "a", iosv.pkg) // verify that we learned the package name and stashed it in the receiver @@ -591,7 +591,7 @@ func TestBailOnVersionBuildMetadata(t *testing.T) { } t.Run("Abort on unorderable build metadata version data", func(t *testing.T) { - _, err := sv.getVersionsFromStandardChannels(&dc) + _, err := sv.getVersionsFromStandardChannels(&dc, buildBundleList(sv)) require.Error(t, err) }) } diff --git a/alpha/template/semver/types.go b/alpha/template/semver/types.go index f170074c3..b0846a64f 100644 --- a/alpha/template/semver/types.go +++ b/alpha/template/semver/types.go @@ -1,19 +1,18 @@ package semver import ( + "context" "io" "github.com/blang/semver/v4" - "github.com/operator-framework/operator-registry/alpha/action/migrations" - "github.com/operator-framework/operator-registry/pkg/image" + "github.com/operator-framework/operator-registry/alpha/declcfg" ) // data passed into this module externally type Template struct { - Data io.Reader - Registry image.Registry - Migrations *migrations.Migrations + Data io.Reader + RenderBundle func(context.Context, string) (*declcfg.DeclarativeConfig, error) } // IO structs -- BEGIN diff --git a/cmd/opm/alpha/template/basic.go b/cmd/opm/alpha/template/basic.go index b1b77a888..4195bd0fe 100644 --- a/cmd/opm/alpha/template/basic.go +++ b/cmd/opm/alpha/template/basic.go @@ -1,6 +1,7 @@ package template import ( + "context" "io" "log" "os" @@ -8,6 +9,7 @@ import ( "github.com/sirupsen/logrus" "github.com/spf13/cobra" + "github.com/operator-framework/operator-registry/alpha/action" "github.com/operator-framework/operator-registry/alpha/action/migrations" "github.com/operator-framework/operator-registry/alpha/declcfg" "github.com/operator-framework/operator-registry/alpha/template/basic" @@ -62,14 +64,23 @@ When FILE is '-' or not provided, the template is read from standard input`, } defer reg.Destroy() - template.Registry = reg - + var m *migrations.Migrations if migrateLevel != "" { - m, err := migrations.NewMigrations(migrateLevel) + m, err = migrations.NewMigrations(migrateLevel) if err != nil { log.Fatal(err) } - template.Migrations = m + } + + template.RenderBundle = func(ctx context.Context, image string) (*declcfg.DeclarativeConfig, error) { + // populate registry, incl any flags from CLI, and enforce only rendering bundle images + r := action.Render{ + Refs: []string{image}, + Registry: reg, + AllowedRefMask: action.RefBundleImage, + Migrations: m, + } + return r.Run(ctx) } // only taking first file argument diff --git a/cmd/opm/alpha/template/semver.go b/cmd/opm/alpha/template/semver.go index e0547a67c..97dccbc6c 100644 --- a/cmd/opm/alpha/template/semver.go +++ b/cmd/opm/alpha/template/semver.go @@ -1,6 +1,7 @@ package template import ( + "context" "fmt" "io" "log" @@ -9,6 +10,7 @@ import ( "github.com/sirupsen/logrus" "github.com/spf13/cobra" + "github.com/operator-framework/operator-registry/alpha/action" "github.com/operator-framework/operator-registry/alpha/action/migrations" "github.com/operator-framework/operator-registry/alpha/declcfg" "github.com/operator-framework/operator-registry/alpha/template/semver" @@ -68,17 +70,27 @@ When FILE is '-' or not provided, the template is read from standard input`, } defer reg.Destroy() - template := semver.Template{ - Data: data, - Registry: reg, - } + var m *migrations.Migrations if migrateLevel != "" { - m, err := migrations.NewMigrations(migrateLevel) + m, err = migrations.NewMigrations(migrateLevel) if err != nil { log.Fatal(err) } - template.Migrations = m } + + template := semver.Template{ + Data: data, + RenderBundle: func(ctx context.Context, ref string) (*declcfg.DeclarativeConfig, error) { + renderer := action.Render{ + Refs: []string{ref}, + Registry: reg, + AllowedRefMask: action.RefBundleImage, + Migrations: m, + } + return renderer.Run(ctx) + }, + } + out, err := template.Render(cmd.Context()) if err != nil { log.Fatalf("semver %q: %v", source, err)