Skip to content
This repository was archived by the owner on Jul 11, 2023. It is now read-only.

k8s/informers: centralize informers to simplify code #4801

Merged

Conversation

keithmattix
Copy link
Contributor

@keithmattix keithmattix commented Jun 9, 2022

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 well

Description:

Testing done:

Affected area:

Functional Area
Control Plane [X]

Please answer the following questions with yes/no.

  1. Does this change contain code from or inspired by another project? no

    • Did you notify the maintainers and provide attribution? N/A
  2. Is this a breaking change? no

  3. Has documentation corresponding to this change been updated in the osm-docs repo (if applicable)? N/A

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]>
Copy link
Contributor

@steeling steeling left a 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 {
Copy link
Contributor

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

Copy link
Contributor Author

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.

Copy link
Contributor

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

Copy link
Contributor Author

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() {
Copy link
Contributor

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 :)

Copy link
Contributor Author

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()).

// 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 {
Copy link
Contributor

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

Copy link
Contributor Author

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)

Copy link
Contributor

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.

Copy link
Contributor Author

@keithmattix keithmattix Jun 9, 2022

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.Stores 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 informer cache.Store.

Signed-off-by: Keith Mattix II <[email protected]>
@keithmattix keithmattix requested a review from steeling June 9, 2022 16:05
Signed-off-by: Keith Mattix II <[email protected]>
}

// WithKubeClient sets the kubeClient for the InformerCollection
func WithKubeClient(kubeClient kubernetes.Interface) InformerCollectionOption {
Copy link
Contributor

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.

Copy link
Contributor Author

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 👍🏾

Copy link
Contributor

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

Copy link
Contributor Author

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 {
Copy link
Contributor

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

@codecov-commenter
Copy link

codecov-commenter commented Jun 9, 2022

Codecov Report

Merging #4801 (a2b6ed3) into main (1e73ba3) will decrease coverage by 0.34%.
The diff coverage is 50.00%.

@@            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     
Flag Coverage Δ
unittests 68.54% <50.00%> (-0.35%) ⬇️

Flags with carried forward coverage won't be shown. Click here to find out more.

Impacted Files Coverage Δ
cmd/osm-controller/osm-controller.go 13.28% <0.00%> (-0.60%) ⬇️
pkg/k8s/types.go 100.00% <ø> (ø)
pkg/k8s/client.go 95.81% <100.00%> (+1.73%) ⬆️
pkg/identity/types.go 76.92% <0.00%> (-23.08%) ⬇️
pkg/service/types.go 62.50% <0.00%> (-4.17%) ⬇️
pkg/ticker/ticker.go 83.33% <0.00%> (-3.85%) ⬇️
pkg/envoy/lds/rbac.go 80.35% <0.00%> (-2.36%) ⬇️
pkg/trafficpolicy/trafficpolicy.go 95.68% <0.00%> (-1.79%) ⬇️
pkg/catalog/ingress.go 96.29% <0.00%> (-0.23%) ⬇️
pkg/errcode/errcode.go 100.00% <0.00%> (ø)
... and 5 more

Continue to review full report at Codecov.

Legend - Click here to learn more
Δ = absolute <relative> (impact), ø = not affected, ? = missing data
Powered by Codecov. Last update 1e73ba3...a2b6ed3. Read the comment docs.

// 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))
Copy link
Member

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.

Suggested change
c.informers.AddEventHandler(osminformers.InformerKeyNamespace, GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker))
c.informers.AddEventHandler(osminformers.NamespaceKey, GetEventHandlerFuncs(nil, nsEventTypes, c.msgBroker))

Copy link
Contributor Author

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)

@shashankram shashankram changed the title Refactor/centralize informers k8s/informers: centralize informers to simplify code Jun 9, 2022
a.Nil(err)
_ = c.informers[Services].GetStore().Add(tc.service)
c := newClient(ic, nil, nil)
_ = store.Add(tc.service)
Copy link
Contributor

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

Copy link
Contributor Author

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).

Copy link
Contributor

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?

Copy link
Contributor Author

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

Copy link
Contributor

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:

  1. Add all the objects to the k8s client, ie testclient.NewSimpleClientset(tc.namespace, ...), then use the internals to ensure that the informer HasSynced.
  2. Expose an add method on the store itself, so you don't need this fake store.
  3. Don't separate these into 2 different packages if you need to rely so heavily on internal implementation details

@keithmattix keithmattix requested a review from steeling June 9, 2022 17:33
Signed-off-by: Keith Mattix II <[email protected]>
Signed-off-by: Keith Mattix II <[email protected]>
@keithmattix keithmattix requested a review from jaellio June 9, 2022 18:27
@jaellio jaellio merged commit 47c06ab into openservicemesh:main Jun 9, 2022
@keithmattix keithmattix deleted the refactor/centralize-informers branch June 9, 2022 18:40
Sign up for free to subscribe to this conversation on GitHub. Already have an account? Sign in.
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants