-
Notifications
You must be signed in to change notification settings - Fork 1.8k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add generate bundle --ignore-if-only-created-at-changed
option
#6419
Changes from all commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
# entries is a list of entries to include in | ||
# release notes and/or the migration guide | ||
entries: | ||
- description: > | ||
Add ability to ignore bundle updates on `generate bundle` command if createdAt timestamp is the only change. | ||
The flag to use is `--ignore-if-only-createdAt`. | ||
|
||
# kind is one of: | ||
# - addition | ||
# - change | ||
# - deprecation | ||
# - removal | ||
# - bugfix | ||
kind: "addition" | ||
|
||
# Is this a breaking change? | ||
breaking: false | ||
|
||
# NOTE: ONLY USE `pull_request_override` WHEN ADDING THIS | ||
# FILE FOR A PREVIOUSLY MERGED PULL_REQUEST! | ||
# | ||
# The generator auto-detects the PR number from the commit | ||
# message in which this file was originally added. | ||
# | ||
# What is the pull request number (without the "#")? | ||
# pull_request_override: 0 | ||
|
||
|
||
# Migration can be defined to automatically add a section to | ||
# the migration guide. This is required for breaking changes. | ||
# migration: | ||
# header: Header text for the migration section | ||
# body: | | ||
# Body of the migration section. This should be formatted as markdown and can | ||
# span multiple lines. | ||
|
||
# Using the YAML string '|' operator means that newlines in this string will | ||
# be honored and interpretted as newlines in the rendered markdown. |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -18,11 +18,11 @@ import ( | |
"fmt" | ||
"io" | ||
"path/filepath" | ||
"reflect" | ||
"strings" | ||
|
||
"github.com/blang/semver/v4" | ||
"github.com/operator-framework/api/pkg/operators/v1alpha1" | ||
operatorsv1alpha1 "github.com/operator-framework/api/pkg/operators/v1alpha1" | ||
"github.com/operator-framework/operator-registry/pkg/lib/bundle" | ||
|
||
"github.com/operator-framework/operator-sdk/internal/generate/clusterserviceversion/bases" | ||
|
@@ -57,10 +57,14 @@ type Generator struct { | |
// {Cluster}Roles to include in a CSV via their Bindings. | ||
ExtraServiceAccounts []string | ||
// RelatedImages are additional images used by the operator. | ||
RelatedImages []operatorsv1alpha1.RelatedImage | ||
RelatedImages []v1alpha1.RelatedImage | ||
|
||
// Func that returns the writer the generated CSV's bytes are written to. | ||
getWriter func() (io.Writer, error) | ||
// Func that returns the reader the previous CSV's bytes are read from. | ||
getReader func() (io.Reader, error) | ||
|
||
ignoreIfOnlyCreatedAtChanged bool | ||
} | ||
|
||
// Option is a function that modifies a Generator. | ||
|
@@ -88,6 +92,22 @@ func WithBundleWriter(dir string) Option { | |
} | ||
} | ||
|
||
// WithBundleGetter sets a Generator's getter to a bundle CSV file under | ||
// <dir>/manifests. | ||
func WithBundleReader(dir string) Option { | ||
return func(g *Generator) error { | ||
fileName := makeCSVFileName(g.OperatorName) | ||
g.getReader = func() (io.Reader, error) { | ||
return bundleReader(dir, fileName) | ||
} | ||
return nil | ||
} | ||
} | ||
|
||
func bundleReader(dir, fileName string) (io.Reader, error) { | ||
return genutil.Open(filepath.Join(dir, bundle.ManifestsDir), fileName) | ||
} | ||
|
||
// WithPackageWriter sets a Generator's writer to a package CSV file under | ||
// <dir>/<version>. | ||
func WithPackageWriter(dir string) Option { | ||
|
@@ -100,6 +120,13 @@ func WithPackageWriter(dir string) Option { | |
} | ||
} | ||
|
||
func WithIgnoreIfOnlyCreatedAtChanged() Option { | ||
return func(g *Generator) error { | ||
g.ignoreIfOnlyCreatedAtChanged = true | ||
return nil | ||
} | ||
} | ||
|
||
// Generate configures the generator with col and opts then runs it. | ||
func (g *Generator) Generate(opts ...Option) (err error) { | ||
for _, opt := range opts { | ||
|
@@ -119,7 +146,33 @@ func (g *Generator) Generate(opts ...Option) (err error) { | |
|
||
// Add extra annotations to csv | ||
g.setAnnotations(csv) | ||
|
||
// If a reader is set, and there is a flag to not update createdAt, then | ||
// set the CSV's createdAt to the previous CSV's createdAt if its the only change. | ||
if g.ignoreIfOnlyCreatedAtChanged && g.getReader != nil { | ||
r, err := g.getReader() | ||
if err != nil { | ||
return err | ||
} | ||
var prevCSV v1alpha1.ClusterServiceVersion | ||
err = genutil.ReadObject(r, &prevCSV) | ||
if err != nil { | ||
return err | ||
} | ||
if prevCSV.ObjectMeta.Annotations != nil && prevCSV.ObjectMeta.Annotations["createdAt"] != "" { | ||
csvWithoutCreatedAtChange := csv.DeepCopy() | ||
// Set WebhookDefinitions if nil to avoid diffing on it | ||
if prevCSV.Spec.WebhookDefinitions == nil { | ||
prevCSV.Spec.WebhookDefinitions = []v1alpha1.WebhookDescription{} | ||
} | ||
if csvWithoutCreatedAtChange.ObjectMeta.Annotations == nil { | ||
csvWithoutCreatedAtChange.ObjectMeta.Annotations = map[string]string{} | ||
} | ||
csvWithoutCreatedAtChange.ObjectMeta.Annotations["createdAt"] = prevCSV.ObjectMeta.Annotations["createdAt"] | ||
Comment on lines
+163
to
+170
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. We do have the code for walking through the input directories, collecting the manifests and unmarshalling it into the CSV in here. We should be able to fetch the createdAt at that step from annotations, and accordingly set it in here. This would avoid us doing the comparisons with webhooks done below. Another option is, once we identify the CSV manifest, we can unmarshal it into a That would make sure that we are performing all operations related to inputs being passed on through flags in place and not read the input CSV multiple times. Wdyt? There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That sounds more efficient. Will have a look. Thanks! |
||
if reflect.DeepEqual(csvWithoutCreatedAtChange, &prevCSV) { | ||
csv = csvWithoutCreatedAtChange | ||
} | ||
} | ||
} | ||
w, err := g.getWriter() | ||
if err != nil { | ||
return err | ||
|
@@ -140,7 +193,7 @@ func (g Generator) setAnnotations(csv *v1alpha1.ClusterServiceVersion) { | |
} | ||
|
||
// generate runs a configured Generator. | ||
func (g *Generator) generate() (base *operatorsv1alpha1.ClusterServiceVersion, err error) { | ||
func (g *Generator) generate() (base *v1alpha1.ClusterServiceVersion, err error) { | ||
if g.Collector == nil { | ||
return nil, fmt.Errorf("cannot generate CSV without a manifests collection") | ||
} | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This shouldn't happen of
overwrite
is set to false. If overwrite is explicitly set to false, andignoreIfOnlyCreatedAtChanged
is passed then we should possibly error out. If not, that defeats the purpose of us trackingcreatedAt
to see the latest changes in bundles. It would be helpful to add that check here.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
+1