-
Notifications
You must be signed in to change notification settings - Fork 276
k8s/informers: centralize informers to simplify code #4801
k8s/informers: centralize informers to simplify code #4801
Conversation
Instead of creating different objects (from 5 different packages) that each manage their own informer collections, create a single object that runs a set of informers. A pointer to this object is then shared with all objects that need it Signed-off-by: Keith Mattix II <[email protected]>
Signed-off-by: Keith Mattix II <[email protected]>
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.
A lot of questions intended as thought experiment, with some concrete suggestions as well
InformerKeyRetry: ic.initRetryMonitor, | ||
} | ||
|
||
if len(ic.selectedInformers) == 0 { |
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.
Consider allowing no informers to run, and defaulting to the raw client calls
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 feel like the informer collection can probably assume that the informers should be used rather than raw client calls.
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.
One of the issues is that (I think) calling Add on the store, only adds the object to the local store, and not to the k8s master. IDK a scenario in which we would actually want to expose that. Removing footguns would be a benefit
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.
So with this proposed implementation, the store is never actually exposed unless you opt-in by passing in a custom store. The previous implementations kept the store exposed for testing purposes (which I definitely think was the right tradeoff given the complexity of timing some of these tests with fake informers), but I agree that removing that possible footgun is preferable which is why I added that custom store option
ic.informers[InformerKeyPod] = informer | ||
} | ||
|
||
func (ic *InformerCollection) initEndpointMonitor() { |
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.
A lot of boilerplate in these functions. Could we clean this up with a map?
ie: for key, client := range ic.informers {
... init and cache informer.
}
You would need a getter on the informer factory though. Again just trying to provoke the train of thought :)
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.
We essentially do that in the NewInformerCollection function; the reason why it's difficult to de-duplicate further is because of the informer factory initialization (e.g. informerFactory.Access().V1alpha3().TrafficTargets().Informer()
).
pkg/k8s/informers/informers.go
Outdated
// WithCustomStore sets the shared store for the informerCollection to reference. | ||
// This store will be used instead of each informer's store. This should typically | ||
// only be used in tests | ||
func WithCustomStores(stores map[InformerKey]cache.Store) InformerCollectionOpt { |
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.
Can you provide an example when we would want to use this in tests?
Does this have improvements over the fakeclient stuff?
If not, I'd say let's take it out, and refactor out the customStore stuff in the informer type below
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.
Yeah this was actually necessary to prevent us from having to run informers and deal with timing issues in tests. Check out client_test.go. GitHub hides it by default (large diff)
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.
Can you expand on those issues? Maybe write a comment on why it's there including technical details.
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.
Added the comment here since GitHub won't update the code showed in the PR comment:
WithCustomStores provides the InformerCollection an set of
cache.Store
s indexed
by InformerKey. This functionality was added for the express purpose of testing
flexibility since the alternative often leads to flaky tests and race conditions
between the time an object is added to a fake clientset and when that object
is actually added to the informercache.Store
.
Signed-off-by: Keith Mattix II <[email protected]>
Signed-off-by: Keith Mattix II <[email protected]>
} | ||
|
||
// WithKubeClient sets the kubeClient for the InformerCollection | ||
func WithKubeClient(kubeClient kubernetes.Interface) InformerCollectionOption { |
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.
Is there ever a place where we pass in a client but don't run the informer? If not, I'd say this could be the signal.
If there is, maybe we could add a bool here, runInformer? Simplifying the API otherwise you could specify an informer to run where the client was never passed in, which would be a bug.
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.
Good call; I've implemented that pattern 👍🏾
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.
a further suggestion: put the init methods in here and get rid of the initInformerMap
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 the map makes initialization cleaner; the alternatives are to re-declare a subset of the map and loop through the keys again or to have each method called sequentially. Happy to discuss further, but I feel like it's probably a preference
InformerKeyRetry: ic.initRetryMonitor, | ||
} | ||
|
||
if len(ic.selectedInformers) == 0 { |
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.
One of the issues is that (I think) calling Add on the store, only adds the object to the local store, and not to the k8s master. IDK a scenario in which we would actually want to expose that. Removing footguns would be a benefit
Signed-off-by: Keith Mattix II <[email protected]>
Codecov Report
@@ Coverage Diff @@
## main #4801 +/- ##
==========================================
- Coverage 68.89% 68.54% -0.35%
==========================================
Files 223 223
Lines 16399 16225 -174
==========================================
- Hits 11298 11122 -176
- Misses 5049 5052 +3
+ Partials 52 51 -1
Flags with carried forward coverage won't be shown. Click here to find out more.
Continue to review full report at Codecov.
|
// Add event handler to informer | ||
nsEventTypes := EventTypes{ | ||
Add: announcements.NamespaceAdded, | ||
Update: announcements.NamespaceUpdated, | ||
Delete: announcements.NamespaceDeleted, | ||
} | ||
c.informers[Namespaces].AddEventHandler(GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker)) | ||
c.informers.AddEventHandler(osminformers.InformerKeyNamespace, GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker)) |
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.
nit: we can drop the prefix InformerKey
from the variable name. Same applies to other informer keys.
c.informers.AddEventHandler(osminformers.InformerKeyNamespace, GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker)) | |
c.informers.AddEventHandler(osminformers.NamespaceKey, GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker)) |
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 prefix is actually a naming convention used in client-go and other packages when creating variables for a type alias. It helps remove ambiguity when searching for ambiguous or overloaded vars (like namespace or pod)
Signed-off-by: Keith Mattix II <[email protected]>
a.Nil(err) | ||
_ = c.informers[Services].GetStore().Add(tc.service) | ||
c := newClient(ic, nil, nil) | ||
_ = store.Add(tc.service) |
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.
So the real store doesn't return the value immediately when doing an add followed by a get?
I thought the only time it wasn't immediate is when doing an add through the actual client, which triggers an async add.
I believe this direct add is synchronous
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.
The direct add is synchronous, but recall that the underlying map of informers is unexported; the tests in k8s/client_test.go don't have access to it (or their stores by extension).
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.
but if it's synchronous why are we instantiating a fake store 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.
Because we can't access the real store from this package
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 see. adding all of this extra stuff to get around this testing seems like a bit of a code smell IMO. I would recommend 1 of 3 things:
- Add all the objects to the k8s client, ie
testclient.NewSimpleClientset(tc.namespace, ...)
, then use the internals to ensure that the informerHasSynced
. - Expose an add method on the store itself, so you don't need this fake store.
- Don't separate these into 2 different packages if you need to rely so heavily on internal implementation details
Signed-off-by: Keith Mattix II <[email protected]>
Signed-off-by: Keith Mattix II <[email protected]>
Instead of creating different objects (from 5 different packages) that
each manage their own informer collections, create a single object that
runs a set of informers. A pointer to this object is then shared with
all objects that need it.
Addresses #4551
This PR only modifies the k8s client; once this is merged, the other 4 clients (SMI, config, configurator, and policy) will be modified to use
InformerCollection
as wellDescription:
Testing done:
Affected area:
Please answer the following questions with yes/no.
Does this change contain code from or inspired by another project? no
Is this a breaking change? no
Has documentation corresponding to this change been updated in the osm-docs repo (if applicable)? N/A