-
Notifications
You must be signed in to change notification settings - Fork 185
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
controllers: add a new controller to handle provider upgrades #3131
base: main
Are you sure you want to change the base?
controllers: add a new controller to handle provider upgrades #3131
Conversation
21eb5de
to
4bdf679
Compare
@rewantsoni please add a meaningful description to the PR |
controllers/storageconsumer/storageconsumer_upgrade_controller.go
Outdated
Show resolved
Hide resolved
controllers/storageconsumer/storageconsumer_upgrade_controller.go
Outdated
Show resolved
Hide resolved
func (r *StorageConsumerUpgradeReconciler) SetupWithManager(mgr ctrl.Manager) error { | ||
|
||
operatorVersionPredicate := predicate.Funcs{ | ||
CreateFunc: func(e event.TypedCreateEvent[client.Object]) bool { | ||
if e.Object == nil { | ||
return false | ||
} | ||
obj := e.Object.(*ocsv1alpha1.StorageConsumer) | ||
return strings.HasPrefix(obj.Status.Client.OperatorVersion, "4.18") && | ||
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | ||
}, | ||
UpdateFunc: func(e event.UpdateEvent) bool { | ||
if e.ObjectOld == nil || e.ObjectNew == nil { | ||
return false | ||
} | ||
newObj := e.ObjectNew.(*ocsv1alpha1.StorageConsumer) | ||
return strings.HasPrefix(newObj.Status.Client.OperatorVersion, "4.18") && | ||
newObj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | ||
}, | ||
} |
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.
You are mixing usage of predicate.Funcs (untyped) with typed events (events.TypedCreateEvent). And then you use to use the type as client.Client
which does not make sense because it is the type you get when you use the untyped versionevent.CreateEvent
.
If you want the predicate to process client.Client
in your handler use:
func (r *StorageConsumerUpgradeReconciler) SetupWithManager(mgr ctrl.Manager) error { | |
operatorVersionPredicate := predicate.Funcs{ | |
CreateFunc: func(e event.TypedCreateEvent[client.Object]) bool { | |
if e.Object == nil { | |
return false | |
} | |
obj := e.Object.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(obj.Status.Client.OperatorVersion, "4.18") && | |
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
UpdateFunc: func(e event.UpdateEvent) bool { | |
if e.ObjectOld == nil || e.ObjectNew == nil { | |
return false | |
} | |
newObj := e.ObjectNew.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(newObj.Status.Client.OperatorVersion, "4.18") && | |
newObj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
} | |
func (r *StorageConsumerUpgradeReconciler) SetupWithManager(mgr ctrl.Manager) error { | |
operatorVersionPredicate := predicate.Funcs{ | |
CreateFunc: func(e event.CreateEvent) bool { | |
if e.Object == nil { | |
return false | |
} | |
obj := e.Object.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(obj.Status.Client.OperatorVersion, "4.18") && | |
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
UpdateFunc: func(e event.UpdateEvent) bool { | |
if e.ObjectOld == nil || e.ObjectNew == nil { | |
return false | |
} | |
newObj := e.ObjectNew.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(newObj.Status.Client.OperatorVersion, "4.18") && | |
newObj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
} |
But becuase you anyway assume an ocsv1alpha1.StorageConsumer entity by casting into it. Then preferably do
func (r *StorageConsumerUpgradeReconciler) SetupWithManager(mgr ctrl.Manager) error { | |
operatorVersionPredicate := predicate.Funcs{ | |
CreateFunc: func(e event.TypedCreateEvent[client.Object]) bool { | |
if e.Object == nil { | |
return false | |
} | |
obj := e.Object.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(obj.Status.Client.OperatorVersion, "4.18") && | |
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
UpdateFunc: func(e event.UpdateEvent) bool { | |
if e.ObjectOld == nil || e.ObjectNew == nil { | |
return false | |
} | |
newObj := e.ObjectNew.(*ocsv1alpha1.StorageConsumer) | |
return strings.HasPrefix(newObj.Status.Client.OperatorVersion, "4.18") && | |
newObj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
} | |
func (r *StorageConsumerUpgradeReconciler) SetupWithManager(mgr ctrl.Manager) error { | |
operatorVersionPredicate := predicate.TypedFuncs[ocsv1alpha1.StorageConsumer]{ | |
CreateFunc: func(e event.TypedCreateEvent[ocsv1alpha1.StorageConsumer]) bool { | |
return strings.HasPrefix(e.obj.Status.Client.OperatorVersion, "4.18") && | |
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
UpdateFunc: func(e event.TypedUpdateEvent[ocsv1alpha1.StorageConsumer]) bool { | |
return strings.HasPrefix(e.newObj.Status.Client.OperatorVersion, "4.18") && | |
newObj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed | |
}, | |
} |
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.
Getting this error:
controllers/storageconsumer/storageconsumer_upgrade_controller.go:60:62: cannot use oldUnProcessedConsumerPredicate (variable of struct type predicate.TypedFuncs["github.com/red-hat-storage/ocs-operator/api/v4/v1alpha1".StorageConsumer]) as predicate.Predicate value in argument to builder.WithPredicates: predicate.TypedFuncs["github.com/red-hat-storage/ocs-operator/api/v4/v1alpha1".StorageConsumer] does not implement predicate.TypedPredicate[client.Object] (wrong type for method Create)
have Create(event.TypedCreateEvent["github.com/red-hat-storage/ocs-operator/api/v4/v1alpha1".StorageConsumer]) bool
want Create(event.TypedCreateEvent[client.Object]) bool
} | ||
obj := e.Object.(*ocsv1alpha1.StorageConsumer) | ||
return strings.HasPrefix(obj.Status.Client.OperatorVersion, "4.18") && | ||
obj.GetAnnotations()[util.StorageConsumer418ProcessedAnnotationKey] != util.StorageConsumer418ProcessedAnnotationProcessed |
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.
I don't get this filter. Why are you trying to filter based on an annotation? and even if you do it does not make sense to look at the value where the mare existence of the annotation is the thing that is important
controllers/storageconsumer/storageconsumer_upgrade_controller.go
Outdated
Show resolved
Hide resolved
{Name: util.GenerateNameForCephBlockPoolSC(storageCluster)}, | ||
{Name: util.GenerateNameForCephFilesystemSC(storageCluster)}, |
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.
Not related to this PR, but why the suffix SC and not StorageClass? the other Generate calls have the full type of the CR at the end: SnapshotClass and GropSnapshotClass
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.
I agree, I will raise a fix to rename all generate to have StorageClass as suffix
{Name: util.GenerateNameForGroupSnapshotClass(storageCluster, util.RbdGroupSnapshotter)}, | ||
{Name: util.GenerateNameForGroupSnapshotClass(storageCluster, util.CephfsGroupSnapshotter)}, | ||
} | ||
util.AddAnnotation(storageConsumer, util.StorageConsumer418ProcessedAnnotationKey, util.StorageConsumer418ProcessedAnnotationProcessed) |
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.
There is no point in having a value for this annotation. Use "" or "true"
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.
I think you decided to use "", but that's come with its own issue of have a need to get a secondary argument when accessing the annotation map to check for existence. While it is not a bad thing, it does make the code more cumbersome.
Let's use strconv.FormatBool(true)
instead
controllers/storageconsumer/storageconsumer_upgrade_controller.go
Outdated
Show resolved
Hide resolved
4bdf679
to
3eb6c0b
Compare
[APPROVALNOTIFIER] This PR is NOT APPROVED This pull-request has been approved by: rewantsoni The full list of commands accepted by this bot can be found here.
Needs approval from an approver in each of these files:
Approvers can indicate their approval by writing |
9455cd5
to
a2e0228
Compare
// For Provider Mode we supported creating one rbdClaim, generating the clientProfile, rns name, secret names | ||
// based on what we decided |
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.
How does a reader understand the sentence based on what we decided
? With comments you should assume the reader have no outside context
rbdStorageRequestHash := getStorageRequestName(string(storageConsumer.UID), rbdClaimName) | ||
rbdNodeSecretName := storageClaimCephCsiSecretName("node", rbdStorageRequestHash) | ||
rbdProvisionerSecretName := storageClaimCephCsiSecretName("provisioner", rbdStorageRequestHash) | ||
md5Sum = md5.Sum([]byte(rbdStorageRequestHash)) |
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.
don't use the same var name again in the same scope
if err := r.Client.Update(ctx, storageConsumer); err != nil { | ||
return reconcile.Result{}, err | ||
} |
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.
There is no point in returning an error if the error you get back is not found
|
||
consumerConfigMapName := fmt.Sprintf("storageconsumer-%v", util.FnvHash(storageConsumer.Name)) | ||
|
||
if storageConsumer.Spec.ResourceNameMappingConfigMap.Name == consumerConfigMapName { |
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.
You should not check for a specific value, but check for having a value.
return reconcile.Result{}, nil | ||
} | ||
|
||
consumerConfigMapName := fmt.Sprintf("storageconsumer-%v", util.FnvHash(storageConsumer.Name)) |
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.
Please calculate this near the creation code of the config map where it is needed
a2e0228
to
354db75
Compare
Signed-off-by: Rewant Soni <[email protected]>
354db75
to
f0343b7
Compare
@rewantsoni: The following tests failed, say
Full PR test history. Your PR dashboard. Instructions for interacting with me using PR comments are available here. If you have questions or suggestions related to my behavior, please file an issue against the kubernetes-sigs/prow repository. I understand the commands that are listed here. |
For Provider Mode update to 4.19, we are creating a new controller
StorageConsumerUpdateController
. This will look at the existing consumers for which the clients are in DF 4.18 and process these consumers to create a configMap and then update the consumer spec to include all the default classes. Once all the changes are made we will add an annotation to these consumers to imply that these are processed, and we can proceed with the reconciliation of these consumers.