diff --git a/alpha/action/migrate.go b/alpha/action/migrate.go index 962cd6489..8122d1648 100644 --- a/alpha/action/migrate.go +++ b/alpha/action/migrate.go @@ -5,6 +5,7 @@ import ( "fmt" "os" + "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" ) @@ -12,7 +13,7 @@ import ( type Migrate struct { CatalogRef string OutputDir string - Level string + Migrations *migrations.Migrations WriteFunc declcfg.WriteFunc FileExt string @@ -29,8 +30,8 @@ func (m Migrate) Run(ctx context.Context) error { } r := Render{ - Refs: []string{m.CatalogRef}, - MigrationLevel: m.Level, + Refs: []string{m.CatalogRef}, + Migrations: m.Migrations, // Only allow catalogs to be migrated. AllowedRefMask: RefSqliteImage | RefSqliteFile | RefDCImage | RefDCDir, diff --git a/alpha/action/migrations/000_bundle_object_to_csv_metadata.go b/alpha/action/migrations/000_bundle_object_to_csv_metadata.go index 5edd2ef5f..d5c6896ea 100644 --- a/alpha/action/migrations/000_bundle_object_to_csv_metadata.go +++ b/alpha/action/migrations/000_bundle_object_to_csv_metadata.go @@ -9,42 +9,39 @@ import ( "github.com/operator-framework/operator-registry/alpha/property" ) -var bundleObjectToCSVMetadata = newMigration( - "bundle-object-to-csv-metadata", - "migrates bundles' `olm.bundle.object` to `olm.csv.metadata`", - func(cfg *declcfg.DeclarativeConfig) error { - convertBundleObjectToCSVMetadata := func(b *declcfg.Bundle) error { - if b.Image == "" || b.CsvJSON == "" { - return nil - } - - var csv v1alpha1.ClusterServiceVersion - if err := json.Unmarshal([]byte(b.CsvJSON), &csv); err != nil { - return err - } - - props := b.Properties[:0] - for _, p := range b.Properties { - switch p.Type { - case property.TypeBundleObject: - // Get rid of the bundle objects - case property.TypeCSVMetadata: - // If this bundle already has a CSV metadata - // property, we won't mutate the bundle at all. - return nil - default: - // Keep all of the other properties - props = append(props, p) - } - } - b.Properties = append(props, property.MustBuildCSVMetadata(csv)) +func bundleObjectToCSVMetadata(cfg *declcfg.DeclarativeConfig) error { + convertBundleObjectToCSVMetadata := func(b *declcfg.Bundle) error { + if b.Image == "" || b.CsvJSON == "" { return nil } - for bi := range cfg.Bundles { - if err := convertBundleObjectToCSVMetadata(&cfg.Bundles[bi]); err != nil { - return err + var csv v1alpha1.ClusterServiceVersion + if err := json.Unmarshal([]byte(b.CsvJSON), &csv); err != nil { + return err + } + + props := b.Properties[:0] + for _, p := range b.Properties { + switch p.Type { + case property.TypeBundleObject: + // Get rid of the bundle objects + case property.TypeCSVMetadata: + // If this bundle already has a CSV metadata + // property, we won't mutate the bundle at all. + return nil + default: + // Keep all of the other properties + props = append(props, p) } } + b.Properties = append(props, property.MustBuildCSVMetadata(csv)) return nil - }) + } + + for bi := range cfg.Bundles { + if err := convertBundleObjectToCSVMetadata(&cfg.Bundles[bi]); err != nil { + return err + } + } + return nil +} diff --git a/alpha/action/migrations/migrations.go b/alpha/action/migrations/migrations.go index 6f4e3a3ed..0c4fc8008 100644 --- a/alpha/action/migrations/migrations.go +++ b/alpha/action/migrations/migrations.go @@ -9,20 +9,33 @@ import ( "github.com/operator-framework/operator-registry/alpha/declcfg" ) +type MigrationToken string + +const ( + invalidMigration string = "" + NoMigrations string = "none" + AllMigrations string = "all" +) + type Migration interface { - Name() string + Token() MigrationToken Help() string Migrate(*declcfg.DeclarativeConfig) error } -func newMigration(name string, help string, fn func(config *declcfg.DeclarativeConfig) error) Migration { - return &simpleMigration{name: name, help: help, fn: fn} +func newMigration(token string, help string, fn func(config *declcfg.DeclarativeConfig) error) Migration { + return &simpleMigration{token: MigrationToken(token), help: help, fn: fn} } type simpleMigration struct { - name string - help string - fn func(*declcfg.DeclarativeConfig) error + token MigrationToken + name string + help string + fn func(*declcfg.DeclarativeConfig) error +} + +func (s simpleMigration) Token() MigrationToken { + return s.token } func (s simpleMigration) Name() string { @@ -41,22 +54,33 @@ type Migrations struct { Migrations []Migration } -func GetLastMigrationName() string { - if len(allMigrations) == 0 { - return "" +func validateName(name string) (MigrationToken, error) { + if name == AllMigrations { + return MigrationToken(AllMigrations), nil } - return allMigrations[len(allMigrations)-1].Name() + for _, migration := range allMigrations { + if migration.Token() == MigrationToken(name) { + return migration.Token(), nil + } + } + return MigrationToken(invalidMigration), fmt.Errorf("unknown migration level %q", name) } // allMigrations represents the migration catalog // the order of these migrations is important var allMigrations = []Migration{ - bundleObjectToCSVMetadata, + newMigration(NoMigrations, "do nothing", func(_ *declcfg.DeclarativeConfig) error { return nil }), + newMigration("bundle-object-to-csv-metadata", "migrates bundles' `olm.bundle.object` to `olm.csv.metadata`", bundleObjectToCSVMetadata), } -func NewMigrations(level string) (*Migrations, error) { - if level == "" { - return &Migrations{}, nil +func NewMigrations(name string) (*Migrations, error) { + if name == AllMigrations { + return &Migrations{Migrations: slices.Clone(allMigrations)}, nil + } + + token, err := validateName(name) + if err != nil { + return nil, err } migrations := slices.Clone(allMigrations) @@ -65,28 +89,28 @@ func NewMigrations(level string) (*Migrations, error) { keep := migrations[:0] for _, migration := range migrations { keep = append(keep, migration) - if migration.Name() == level { + if migration.Token() == token { found = true break } } if !found { - return nil, fmt.Errorf("unknown migration level %q", level) + return nil, fmt.Errorf("unknown migration level %q", name) } return &Migrations{Migrations: keep}, nil } func HelpText() string { var help strings.Builder - help.WriteString(" The migrator will run all migrations up to and including the selected level.\n\n") - help.WriteString(" Available migrators:\n") + help.WriteString("\nThe migrator will run all migrations up to and including the selected level.\n\n") + help.WriteString("Available migrators:\n") if len(allMigrations) == 0 { help.WriteString(" (no migrations available in this version)\n") } - tabber := tabwriter.NewWriter(&help, 20, 30, 1, '\t', tabwriter.AlignRight) + tabber := tabwriter.NewWriter(&help, 0, 0, 1, ' ', 0) for _, migration := range allMigrations { - fmt.Fprintf(tabber, " - %s\t%s\n", migration.Name(), migration.Help()) + fmt.Fprintf(tabber, " - %s\t: %s\n", migration.Token(), migration.Help()) } tabber.Flush() return help.String() diff --git a/alpha/action/render.go b/alpha/action/render.go index 75ea58a07..798ffb5e5 100644 --- a/alpha/action/render.go +++ b/alpha/action/render.go @@ -54,8 +54,8 @@ type Render struct { Refs []string Registry image.Registry AllowedRefMask RefType - MigrationLevel string ImageRefTemplate *template.Template + Migrations *migrations.Migrations skipSqliteDeprecationLog bool } @@ -88,7 +88,7 @@ func (r Render) Run(ctx context.Context) (*declcfg.DeclarativeConfig, error) { }) } - if err := migrate(cfg, r.MigrationLevel); err != nil { + if err := r.migrate(cfg); err != nil { return nil, fmt.Errorf("migrate: %v", err) } @@ -414,18 +414,12 @@ func moveBundleObjectsToEndOfPropertySlices(cfg *declcfg.DeclarativeConfig) { } } -func migrate(cfg *declcfg.DeclarativeConfig, migrateLevel string) error { - mobj, err := migrations.NewMigrations(migrateLevel) - if err != nil { - return err - } - - err = mobj.Migrate(cfg) - if err != nil { - return err +func (r Render) migrate(cfg *declcfg.DeclarativeConfig) error { + // If there are no migrations, do nothing. + if r.Migrations == nil { + return nil } - - return nil + return r.Migrations.Migrate(cfg) } func combineConfigs(cfgs []declcfg.DeclarativeConfig) *declcfg.DeclarativeConfig { diff --git a/alpha/action/render_test.go b/alpha/action/render_test.go index a00721812..9428be8ad 100644 --- a/alpha/action/render_test.go +++ b/alpha/action/render_test.go @@ -20,6 +20,7 @@ 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/alpha/property" "github.com/operator-framework/operator-registry/pkg/containertools" @@ -29,9 +30,6 @@ import ( "github.com/operator-framework/operator-registry/pkg/sqlite" ) -// aligns with supported migrations in alpha.action.migrations -const migrationLevelCSVMetadata = "bundle-object-to-csv-metadata" - func TestRender(t *testing.T) { type spec struct { name string @@ -75,13 +73,18 @@ func TestRender(t *testing.T) { image.SimpleReference("test.registry/foo-operator/foo-bundle:v0.2.0"): "testdata/foo-bundle-v0.2.0", } assert.NoError(t, generateSqliteFile(dbFile, imageMap)) + allMigrations, err := migrations.NewMigrations(migrations.AllMigrations) + require.NoError(t, err) + noMigrations, err := migrations.NewMigrations(migrations.NoMigrations) + require.NoError(t, err) specs := []spec{ { name: "Success/SqliteIndexImage", render: action.Render{ - Refs: []string{"test.registry/foo-operator/foo-index-sqlite:v0.2.0"}, - Registry: reg, + Refs: []string{"test.registry/foo-operator/foo-index-sqlite:v0.2.0"}, + Registry: reg, + Migrations: noMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Packages: []declcfg.Package{ @@ -172,9 +175,9 @@ func TestRender(t *testing.T) { { name: "Success/SqliteIndexImageCSVMigration", render: action.Render{ - Refs: []string{"test.registry/foo-operator/foo-index-sqlite:v0.2.0"}, - Registry: reg, - MigrationLevel: migrationLevelCSVMetadata, + Refs: []string{"test.registry/foo-operator/foo-index-sqlite:v0.2.0"}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Packages: []declcfg.Package{ @@ -355,9 +358,9 @@ func TestRender(t *testing.T) { { name: "Success/SqliteFileMigration", render: action.Render{ - Refs: []string{dbFile}, - Registry: reg, - MigrationLevel: migrationLevelCSVMetadata, + Refs: []string{dbFile}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Packages: []declcfg.Package{ @@ -642,9 +645,9 @@ func TestRender(t *testing.T) { { name: "Success/DeclcfgImageMigrate", render: action.Render{ - Refs: []string{"test.registry/foo-operator/foo-index-declcfg:v0.2.0"}, - MigrationLevel: migrationLevelCSVMetadata, - Registry: reg, + Refs: []string{"test.registry/foo-operator/foo-index-declcfg:v0.2.0"}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Packages: []declcfg.Package{ @@ -739,9 +742,9 @@ func TestRender(t *testing.T) { { name: "Success/DeclcfgDirectoryMigrate", render: action.Render{ - Refs: []string{"testdata/foo-index-v0.2.0-declcfg"}, - MigrationLevel: migrationLevelCSVMetadata, - Registry: reg, + Refs: []string{"testdata/foo-index-v0.2.0-declcfg"}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Packages: []declcfg.Package{ @@ -886,9 +889,9 @@ func TestRender(t *testing.T) { { name: "Success/BundleImageMigration", render: action.Render{ - Refs: []string{"test.registry/foo-operator/foo-bundle:v0.2.0"}, - Registry: reg, - MigrationLevel: migrationLevelCSVMetadata, + Refs: []string{"test.registry/foo-operator/foo-bundle:v0.2.0"}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Bundles: []declcfg.Bundle{ @@ -981,9 +984,9 @@ func TestRender(t *testing.T) { { name: "Success/BundleImageWithNoCSVRelatedImagesMigration", render: action.Render{ - Refs: []string{"test.registry/foo-operator/foo-bundle-no-csv-related-images:v0.2.0"}, - Registry: reg, - MigrationLevel: migrationLevelCSVMetadata, + Refs: []string{"test.registry/foo-operator/foo-bundle-no-csv-related-images:v0.2.0"}, + Registry: reg, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Bundles: []declcfg.Bundle{ @@ -1077,7 +1080,7 @@ func TestRender(t *testing.T) { Refs: []string{"testdata/foo-bundle-v0.2.0"}, ImageRefTemplate: template.Must(template.New("imageRef").Parse("test.registry/{{.Package}}-operator/{{.Package}}:v{{.Version}}")), Registry: reg, - MigrationLevel: migrationLevelCSVMetadata, + Migrations: allMigrations, }, expectCfg: &declcfg.DeclarativeConfig{ Bundles: []declcfg.Bundle{ diff --git a/alpha/template/basic/basic.go b/alpha/template/basic/basic.go index fc467162e..0456dc151 100644 --- a/alpha/template/basic/basic.go +++ b/alpha/template/basic/basic.go @@ -9,6 +9,7 @@ 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" ) @@ -16,8 +17,8 @@ import ( const schema string = "olm.template.basic" type Template struct { - Registry image.Registry - MigrationLevel string + Registry image.Registry + Migrations *migrations.Migrations } type BasicTemplate struct { @@ -60,7 +61,7 @@ func (t Template) Render(ctx context.Context, reader io.Reader) (*declcfg.Declar r := action.Render{ Registry: t.Registry, AllowedRefMask: action.RefBundleImage, - MigrationLevel: t.MigrationLevel, + Migrations: t.Migrations, } for _, b := range cfg.Bundles { diff --git a/alpha/template/semver/semver.go b/alpha/template/semver/semver.go index 67b71b9e1..05135a318 100644 --- a/alpha/template/semver/semver.go +++ b/alpha/template/semver/semver.go @@ -35,7 +35,7 @@ func (t Template) Render(ctx context.Context) (*declcfg.DeclarativeConfig, error AllowedRefMask: action.RefBundleImage, Refs: []string{b}, Registry: t.Registry, - MigrationLevel: t.MigrationLevel, + Migrations: t.Migrations, } c, err := r.Run(ctx) if err != nil { diff --git a/alpha/template/semver/types.go b/alpha/template/semver/types.go index 458252321..f170074c3 100644 --- a/alpha/template/semver/types.go +++ b/alpha/template/semver/types.go @@ -5,14 +5,15 @@ import ( "github.com/blang/semver/v4" + "github.com/operator-framework/operator-registry/alpha/action/migrations" "github.com/operator-framework/operator-registry/pkg/image" ) // data passed into this module externally type Template struct { - Data io.Reader - Registry image.Registry - MigrationLevel string + Data io.Reader + Registry image.Registry + Migrations *migrations.Migrations } // IO structs -- BEGIN diff --git a/cmd/opm/alpha/template/basic.go b/cmd/opm/alpha/template/basic.go index 3729df960..b1b77a888 100644 --- a/cmd/opm/alpha/template/basic.go +++ b/cmd/opm/alpha/template/basic.go @@ -16,7 +16,8 @@ import ( func newBasicTemplateCmd() *cobra.Command { var ( - template basic.Template + template basic.Template + migrateLevel string ) cmd := &cobra.Command{ Use: "basic basic-template-file", @@ -63,6 +64,14 @@ When FILE is '-' or not provided, the template is read from standard input`, template.Registry = reg + if migrateLevel != "" { + m, err := migrations.NewMigrations(migrateLevel) + if err != nil { + log.Fatal(err) + } + template.Migrations = m + } + // only taking first file argument cfg, err := template.Render(cmd.Context(), data) if err != nil { @@ -75,7 +84,7 @@ When FILE is '-' or not provided, the template is read from standard input`, }, } - cmd.Flags().StringVar(&template.MigrationLevel, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) + cmd.Flags().StringVar(&migrateLevel, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) return cmd } diff --git a/cmd/opm/alpha/template/semver.go b/cmd/opm/alpha/template/semver.go index 7caf93036..e0547a67c 100644 --- a/cmd/opm/alpha/template/semver.go +++ b/cmd/opm/alpha/template/semver.go @@ -72,6 +72,13 @@ When FILE is '-' or not provided, the template is read from standard input`, Data: data, Registry: reg, } + if migrateLevel != "" { + m, err := migrations.NewMigrations(migrateLevel) + if err != nil { + log.Fatal(err) + } + template.Migrations = m + } out, err := template.Render(cmd.Context()) if err != nil { log.Fatalf("semver %q: %v", source, err) diff --git a/cmd/opm/migrate/cmd.go b/cmd/opm/migrate/cmd.go index 501763090..edf496a33 100644 --- a/cmd/opm/migrate/cmd.go +++ b/cmd/opm/migrate/cmd.go @@ -14,8 +14,9 @@ import ( func NewCmd() *cobra.Command { var ( - migrate action.Migrate - output string + migrate action.Migrate + migrateLevel string + output string ) cmd := &cobra.Command{ Use: "migrate ", @@ -43,6 +44,14 @@ parsers that assume that a file contains exactly one valid JSON object. log.Fatalf("invalid --output value %q, expected (json|yaml)", output) } + if migrateLevel != "" { + m, err := migrations.NewMigrations(migrateLevel) + if err != nil { + log.Fatal(err) + } + migrate.Migrations = m + } + logrus.Infof("rendering index %q as file-based catalog", migrate.CatalogRef) if err := migrate.Run(cmd.Context()); err != nil { logrus.New().Fatal(err) @@ -52,7 +61,7 @@ parsers that assume that a file contains exactly one valid JSON object. }, } cmd.Flags().StringVarP(&output, "output", "o", "json", "Output format (json|yaml)") - cmd.Flags().StringVar(&migrate.Level, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) + cmd.Flags().StringVar(&migrateLevel, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) return cmd } diff --git a/cmd/opm/render/cmd.go b/cmd/opm/render/cmd.go index ebc8bb69d..683d11a1c 100644 --- a/cmd/opm/render/cmd.go +++ b/cmd/opm/render/cmd.go @@ -22,7 +22,8 @@ func NewCmd(showAlphaHelp bool) *cobra.Command { output string imageRefTemplate string - deprecatedMigrateFlag bool + oldMigrateAllFlag bool + migrateLevel string ) cmd := &cobra.Command{ Use: "render [catalog-image | catalog-directory | bundle-image | bundle-directory | sqlite-file]...", @@ -67,9 +68,16 @@ database files. } // if the deprecated flag was used, set the level explicitly to the last migration to perform all migrations - if deprecatedMigrateFlag { - render.MigrationLevel = migrations.GetLastMigrationName() + var m *migrations.Migrations + if oldMigrateAllFlag { + m, err = migrations.NewMigrations(migrations.AllMigrations) + } else if migrateLevel != "" { + m, err = migrations.NewMigrations(migrateLevel) } + if err != nil { + log.Fatal(err) + } + render.Migrations = m cfg, err := render.Run(cmd.Context()) if err != nil { @@ -83,9 +91,8 @@ database files. } cmd.Flags().StringVarP(&output, "output", "o", "json", "Output format of the streamed file-based catalog objects (json|yaml)") - cmd.Flags().StringVar(&render.MigrationLevel, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) - cmd.Flags().BoolVar(&deprecatedMigrateFlag, "migrate", false, "Perform migrations on the rendered FBC") - cmd.Flags().MarkDeprecated("migrate", "use --migrate-level instead") + cmd.Flags().StringVar(&migrateLevel, "migrate-level", "", "Name of the last migration to run (default: none)\n"+migrations.HelpText()) + cmd.Flags().BoolVar(&oldMigrateAllFlag, "migrate", false, "Perform all available schema migrations on the rendered FBC") cmd.MarkFlagsMutuallyExclusive("migrate", "migrate-level") // Alpha flags