Skip to content
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

Non Watched Namespace Labeled selectors #515

Open
wants to merge 106 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
106 commits
Select commit Hold shift + click to select a range
c3c1bfa
initial commit before code generation
jackstine Aug 17, 2022
9d2a486
solving generator problems
jackstine Aug 18, 2022
a8a66dd
updated the snahost emitter template, it now generates code.
jackstine Aug 18, 2022
ed99470
comment with added
jackstine Aug 18, 2022
acebcb8
update to testing snapshot emitter
jackstine Aug 18, 2022
c466f4d
added code for testing snapshot emitter
jackstine Aug 22, 2022
0f6b483
updated so that ClusterScoped resource snapshot emitters will work
jackstine Aug 22, 2022
16b5232
update the snapshot_emitter to call out to kube for the namespaces
jackstine Aug 22, 2022
e64c8f1
updates to testing and snapshot emitter
jackstine Aug 23, 2022
670cf7e
update for interface for ResourceNamespaces
jackstine Aug 25, 2022
95d8548
updated snapshot_emitter to filter namespaces by labels, and not reso…
jackstine Aug 25, 2022
86bca00
added client interface
jackstine Aug 25, 2022
90b86d2
left out some things
jackstine Aug 25, 2022
26216b5
get interface to work with expression selectors
jackstine Aug 25, 2022
0ef7be9
Merge refs/heads/master into labeled-selectors
soloio-bulldozer[bot] Aug 26, 2022
c829254
got it passing tests for namespaced labels
jackstine Aug 26, 2022
afc7fd2
Merge branch 'labeled-selectors' of ssh://github.com/solo-io/solo-kit…
jackstine Aug 26, 2022
b2ce0cf
snapshot does not contain deleted namespace resources
jackstine Aug 26, 2022
4cf797d
pass CI for non ClusterScoped
jackstine Aug 26, 2022
765bfc5
adding kube client resource namespace lister
jackstine Aug 29, 2022
36bccb7
added in map for watched namespaces
jackstine Aug 29, 2022
af6eb5d
updated snapshot emitter, delt with err channels and with deleting na…
jackstine Aug 30, 2022
644a6f9
clean up and reviews
jackstine Aug 30, 2022
4c4192f
add missing file generated
jackstine Aug 30, 2022
5c53707
change variable name
jackstine Aug 30, 2022
ac6124f
update with comments, and nits
jackstine Aug 30, 2022
3f60056
moving some functions
jackstine Aug 30, 2022
24ded9e
small change
jackstine Aug 30, 2022
8bd5ec5
closing the channels
jackstine Aug 30, 2022
2a9cf1c
updated the tests
jackstine Aug 30, 2022
20770ed
fix kube event loop error
jackstine Aug 30, 2022
835b2f9
update changelog
jackstine Aug 30, 2022
4c86983
to pass CI retro 1
jackstine Aug 30, 2022
62de6b8
fix
jackstine Aug 31, 2022
0d847d1
fix
jackstine Aug 31, 2022
feb1c79
race conditions, still working on fixing issues with deleted namespaces
jackstine Aug 31, 2022
c2377cf
fix issue with non Expression Selector emitters updating when namespa…
jackstine Sep 1, 2022
b806cfc
update fix problem with namespaces getting terminated, but not deleted
jackstine Sep 1, 2022
8587388
refactored the test code, to make it more easier to manage
jackstine Sep 1, 2022
148dc61
missing build solokit
jackstine Sep 1, 2022
576f2aa
fix issue with name5
jackstine Sep 1, 2022
1a0989a
fix name5
jackstine Sep 1, 2022
3aa046c
v1 alpha
jackstine Sep 1, 2022
f09e7e6
fix problems with parallel tests
jackstine Sep 1, 2022
217108b
update to fix notifications about snapshots
jackstine Sep 1, 2022
a1fb027
missing file updates
jackstine Sep 1, 2022
f1db36e
add updated go build comments
jackstine Sep 1, 2022
f6d374f
Merge ssh://github.com/solo-io/solo-kit into labeledSelectors
jackstine Sep 9, 2022
9aafe60
updated the controller to allow adding informers
jackstine Sep 11, 2022
e7371ca
register new namespace to cache
jackstine Sep 12, 2022
d3c5196
register new namespaces for the Resoruce Client Shared Informer Factory
jackstine Sep 12, 2022
ce03b28
Merge branch 'cache-changes' into labeled-selectors
jackstine Sep 12, 2022
1ae4371
added in Register Namespace to the Resouce client interface.
jackstine Sep 12, 2022
5e9f466
update resource clients
jackstine Sep 13, 2022
b72998b
Merge refs/heads/master into labeled-selectors
soloio-bulldozer[bot] Sep 13, 2022
cb7159b
add rc template
jackstine Sep 13, 2022
e2b5868
Merge branch 'labeled-selectors' of ssh://github.com/solo-io/solo-kit…
jackstine Sep 13, 2022
d05edec
add in once functionality to register calls
jackstine Sep 13, 2022
0e2db70
rcf must start() if we are registering namespaces to it
jackstine Sep 13, 2022
2ad5399
possible race condition fix
jackstine Sep 13, 2022
e9c3539
fix registering issue
jackstine Sep 13, 2022
7af3b64
fix reconciler
jackstine Sep 13, 2022
7b1e280
change test so that the kube cache does not panic
jackstine Sep 13, 2022
b334e28
update the client interface
jackstine Sep 13, 2022
f3597e0
better mutex control
jackstine Sep 13, 2022
6988153
fix race condition in test
jackstine Sep 13, 2022
0641785
add comment
jackstine Sep 13, 2022
2f06f41
allow new namespaces in the kube resouce client
jackstine Sep 13, 2022
57873d6
clean up
jackstine Sep 13, 2022
b9cbb91
added in a test for when namespaces are deleted
jackstine Sep 13, 2022
8bb2c03
fix cache test
jackstine Sep 13, 2022
c074097
fix cache test again
jackstine Sep 13, 2022
f3aa04a
Adding changelog file to new location
Sep 14, 2022
ea8b754
Deleting changelog file from old location
Sep 14, 2022
9894a78
Merge branch 'master' into labeled-selectors
jackstine Sep 15, 2022
39072d5
fix merge conflict issue
jackstine Sep 15, 2022
50578c7
finish todos, add logs for registered namespaces, fixed issue with Re…
jackstine Sep 16, 2022
7b30788
update tests
jackstine Sep 16, 2022
24c3941
fixed issue with test, and cleaned up tests a little
jackstine Sep 19, 2022
93642d1
fix error with kube emitter
jackstine Sep 19, 2022
a047e22
Adding changelog file to new location
Sep 21, 2022
2353077
Deleting changelog file from old location
Sep 21, 2022
f8e908c
Kube Cache now requires a boolean for creating the namespace lister
jackstine Sep 21, 2022
ed44661
Merge branch 'labeled-selectors' of ssh://github.com/solo-io/solo-kit…
jackstine Sep 21, 2022
57e4976
Merge branch 'master' into labeled-selectors
jackstine Sep 21, 2022
8d91464
update to fix merge conflict
jackstine Sep 21, 2022
4c7db7f
when namespaces is set to "" then create the namespace lister
jackstine Sep 21, 2022
1c1fa6c
condition on namespace lister
jackstine Sep 21, 2022
66ddd96
delete failing mock_custom spec hash resource
jackstine Sep 22, 2022
eee5ed2
some touch ups. Errors are now logged in the list and watch functions…
jackstine Sep 22, 2022
53176bf
move changelog
jackstine Sep 22, 2022
b6afc93
Merge refs/heads/master into labeled-selectors
soloio-bulldozer[bot] Sep 28, 2022
548a0c9
Merge refs/heads/master into labeled-selectors
soloio-bulldozer[bot] Sep 29, 2022
d2f6b4d
Merge refs/heads/master into labeled-selectors
soloio-bulldozer[bot] Sep 29, 2022
c3b1510
Adding changelog file to new location
Feb 13, 2023
879f457
Deleting changelog file from old location
Feb 13, 2023
3e2718b
Adding changelog file to new location
May 19, 2023
26bac9d
Deleting changelog file from old location
May 19, 2023
e6f69d4
Adding changelog file to new location
Jul 21, 2023
c0ea174
Deleting changelog file from old location
Jul 21, 2023
f8db291
Adding changelog file to new location
Dec 5, 2023
ebcbbd9
Deleting changelog file from old location
Dec 5, 2023
9f7df5e
Adding changelog file to new location
May 28, 2024
5100636
Deleting changelog file from old location
May 28, 2024
9f20d56
Adding changelog file to new location
Sep 11, 2024
b560d27
Deleting changelog file from old location
Sep 11, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
227 changes: 212 additions & 15 deletions pkg/code-generator/codegen/templates/snapshot_emitter_template.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ var ResourceGroupEmitterTemplate = template.Must(template.New("resource_group_em
{{- $clients := (join_str_slice $clients ", ") }}

import (
"fmt"
"bytes"
"sync"
"time"

Expand All @@ -34,6 +34,9 @@ import (

"github.com/solo-io/go-utils/errutils"
"github.com/solo-io/go-utils/contextutils"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"
kubewatch "k8s.io/apimachinery/pkg/watch"
)

{{ $emitter_prefix := (print (snake .Name) "/emitter") }}
Expand Down Expand Up @@ -159,9 +162,24 @@ func (c *{{ lower_camel .GoName }}Emitter) Snapshots(watchNamespaces []string, o
}

errs := make(chan error)
hasWatchedNamespaces := len(watchNamespaces) > 1 || (len(watchNamespaces) == 1 && watchNamespaces[0] != "")
watchNamespacesIsEmpty := ! hasWatchedNamespaces
var done sync.WaitGroup
ctx := opts.Ctx

// if we are watching namespaces, then we do not want to fitler any of the
// resources in when listing or watching
// TODO-JAKE not sure if we want to get rid of the Selector in the
// ListOpts here. the reason that we might want to is because we no
// longer allow selectors, unless it is on a unwatched namespace.
watchedNamespacesListOptions := clients.ListOpts{Ctx: opts.Ctx}
watchedNamespacesWatchOptions := clients.WatchOpts{Ctx: opts.Ctx}
if watchNamespacesIsEmpty {
// if the namespaces that we are watching is empty, then we want to apply
// the expression Selectors to all the namespaces.
watchedNamespacesListOptions.ExpressionSelector = opts.ExpressionSelector
watchedNamespacesWatchOptions.ExpressionSelector = opts.ExpressionSelector
}

{{- range .Resources}}
/* Create channel for {{ .Name }} */
Expand All @@ -171,32 +189,32 @@ func (c *{{ lower_camel .GoName }}Emitter) Snapshots(watchNamespaces []string, o
namespace string
}
{{ lower_camel .Name }}Chan := make(chan {{ lower_camel .Name }}ListWithNamespace)

var initial{{ upper_camel .Name }}List {{ .ImportPrefix }}{{ .Name }}List{{- end }}

{{- end}}
var initial{{ upper_camel .Name }}List {{ .ImportPrefix }}{{ .Name }}List
{{- end }}
{{- end }}

currentSnapshot := {{ .GoName }}Snapshot{}

{{- range .Resources}}
{{- if not .ClusterScoped }}
{{ lower_camel .PluralName }}ByNamespace := make(map[string]{{ .ImportPrefix }}{{ .Name }}List)
{{- end }}
{{- end }}
{{- range .Resources}}
{{- if not .ClusterScoped }}
{{ lower_camel .PluralName }}ByNamespace := make(map[string]{{ .ImportPrefix }}{{ .Name }}List)
{{- end }}
{{- end }}

// watched namespaces
for _, namespace := range watchNamespaces {
{{- range .Resources}}
{{- if (not .ClusterScoped) }}
/* Setup namespaced watch for {{ .Name }} */
{
{{ lower_camel .PluralName }}, err := c.{{ lower_camel .Name }}.List(namespace, clients.ListOpts{Ctx: opts.Ctx, Selector: opts.Selector})
{{ lower_camel .PluralName }}, err := c.{{ lower_camel .Name }}.List(namespace, watchedNamespacesListOptions)
if err != nil {
return nil, nil, errors.Wrapf(err, "initial {{ .Name }} list")
}
initial{{ upper_camel .Name }}List = append(initial{{ upper_camel .Name }}List, {{ lower_camel .PluralName }}...)
{{ lower_camel .PluralName }}ByNamespace[namespace] = {{ lower_camel .PluralName }}
}
{{ lower_camel .Name }}NamespacesChan, {{ lower_camel .Name }}Errs, err := c.{{ lower_camel .Name }}.Watch(namespace, opts)
{{ lower_camel .Name }}NamespacesChan, {{ lower_camel .Name }}Errs, err := c.{{ lower_camel .Name }}.Watch(namespace, watchedNamespacesWatchOptions)
if err != nil {
return nil, nil, errors.Wrapf(err, "starting {{ .Name }} watch")
}
Expand All @@ -208,8 +226,7 @@ func (c *{{ lower_camel .GoName }}Emitter) Snapshots(watchNamespaces []string, o
}(namespace)

{{- end }}
{{- end}}

{{- end }}
/* Watch for changes and update snapshot */
go func(namespace string) {
for {
Expand All @@ -233,9 +250,189 @@ func (c *{{ lower_camel .GoName }}Emitter) Snapshots(watchNamespaces []string, o
}
}(namespace)
}
if hasWatchedNamespaces && opts.ExpressionSelector != "" {
// watch resources using non-watched namespaces. With these namespaces we
// will watch only those that are filted using the label selectors defined
// by Expression Selectors

// first get the renaiming namespaces
var k kubernetes.Interface
excludeNamespacesFieldDesciptors := ""

var buffer bytes.Buffer
for i, ns := range watchNamespaces {
buffer.WriteString("metadata.namespace!=")
buffer.WriteString(ns)
if i < len(watchNamespaces)-1 {
buffer.WriteByte(',')
}
}
excludeNamespacesFieldDesciptors = buffer.String()
jackstine marked this conversation as resolved.
Show resolved Hide resolved

namespacesResources,err := k.CoreV1().Namespaces().List(ctx, metav1.ListOptions{FieldSelector: excludeNamespacesFieldDesciptors})
if err != nil {
return nil, nil, err
}
allOtherNamespaces := make([]string, len(namespacesResources.Items))
for _, ns := range namespacesResources.Items {
allOtherNamespaces = append(allOtherNamespaces, ns.Namespace)
}

// nonWatchedNamespaces
// REFACTOR
for _, namespace := range allOtherNamespaces {
{{- range .Resources }}
{{- if (not .ClusterScoped) }}
/* Setup namespaced watch for {{ upper_camel .Name }} */
{
{{ lower_camel .PluralName }}, err := c.{{ lower_camel .Name }}.List(namespace, clients.ListOpts{Ctx: opts.Ctx, ExpressionSelector: opts.ExpressionSelector})
if err != nil {
return nil, nil, errors.Wrapf(err, "initial {{ upper_camel .Name }} list")
}
initial{{ upper_camel .Name }}List = append(initial{{ upper_camel .Name }}List,{{ lower_camel .PluralName }}...)
{{ lower_camel .PluralName }}ByNamespace[namespace] = {{ lower_camel .PluralName }}
}
{{ lower_camel .Name }}NamespacesChan, {{ lower_camel .Name }}Errs, err := c.{{ lower_camel .Name }}.Watch(namespace, opts)
if err != nil {
return nil, nil, errors.Wrapf(err, "starting {{ upper_camel .Name }} watch")
}

done.Add(1)
go func(namespace string) {
defer done.Done()
errutils.AggregateErrs(ctx, errs, {{ lower_camel .Name }}Errs, namespace+"-{{ lower_camel .PluralName }}")
}(namespace)
{{- end }}
{{- end }}
/* Watch for changes and update snapshot */
go func(namespace string) {
for {
select {
case <-ctx.Done():
return
{{- range .Resources }}
{{- if (not .ClusterScoped) }}
case {{ lower_camel .Name }}List, ok := <-{{ lower_camel .Name }}NamespacesChan:
if !ok {
return
}
select {
case <-ctx.Done():
return
case {{ lower_camel .Name }}Chan <- {{ lower_camel .Name }}ListWithNamespace{list: {{ lower_camel .Name }}List, namespace: namespace}:
}
{{- end }}
{{- end }}
}
}
}(namespace)
}
// create watch on all namespaces, so that we can add resources from new namespaces
namespaceWatch,err := k.CoreV1().Namespaces().Watch(opts.Ctx, metav1.ListOptions{FieldSelector: excludeNamespacesFieldDesciptors})
if err != nil {
return nil, nil, err
}

go func() {
for {
select {
case <-ctx.Done():
return
case event, ok := <-namespaceWatch.ResultChan():
if !ok {
return
}
switch event.Type {
case kubewatch.Error:
errs <- errors.Errorf("receiving namespace event", event)
default:
namespacesResources, err := k.CoreV1().Namespaces().List(opts.Ctx, metav1.ListOptions{FieldSelector: excludeNamespacesFieldDesciptors})
if err != nil {
errs <- errors.Wrapf(err, "listing the namespace resources")
}

hit := false
newNamespaces := []string{}

for _,item := range namespacesResources.Items {
namespace := item.Namespace
{{- range .Resources }}
{{- if (not .ClusterScoped) }}
_, hit = {{ lower_camel .PluralName }}ByNamespace[namespace]
if ! hit {
newNamespaces = append(newNamespaces, namespace)
continue
}
{{- end }}
{{- end }}
}
if hit {
// add a watch for all the new namespaces
// REFACTOR
for _, namespace := range newNamespaces {
{{- range .Resources }}
{{- if (not .ClusterScoped) }}
/* Setup namespaced watch for {{ upper_camel .Name }} for new namespace */
{
{{ lower_camel .PluralName }}, err := c.{{ lower_camel .Name }}.List(namespace, clients.ListOpts{Ctx: opts.Ctx, ExpressionSelector: opts.ExpressionSelector})
if err != nil {
// INFO-JAKE not sure if we want to do something else
// but since this is occuring in async I think it should be fine
errs <- errors.Wrapf(err, "initial new namespace {{ upper_camel .Name }} list")
continue
}
{{ lower_camel .PluralName }}ByNamespace[namespace] = {{ lower_camel .PluralName }}
}
{{ lower_camel .Name }}NamespacesChan, {{ lower_camel .Name }}Errs, err := c.{{ lower_camel .Name }}.Watch(namespace, opts)
if err != nil {
// INFO-JAKE is this what we really want to do when there is an error?
errs <- errors.Wrapf(err, "starting new namespace {{ upper_camel .Name }} watch")
continue
}

// INFO-JAKE I think this is appropriate, becasue
// we want to watch the errors coming off the namespace
done.Add(1)
go func(namespace string) {
defer done.Done()
errutils.AggregateErrs(ctx, errs, {{ lower_camel .Name }}Errs, namespace+"-new-namespace-{{ lower_camel .PluralName }}")
}(namespace)
{{- end }}
{{- end }}
/* Watch for changes and update snapshot */
// REFACTOR
go func(namespace string) {
for {
select {
case <-ctx.Done():
return
{{- range .Resources }}
{{- if (not .ClusterScoped) }}
case {{ lower_camel .Name }}List, ok := <-{{ lower_camel .Name }}NamespacesChan:
if !ok {
return
}
select {
case <-ctx.Done():
return
case {{ lower_camel .Name }}Chan <- {{ lower_camel .Name }}ListWithNamespace{list: {{ lower_camel .Name }}List, namespace: namespace}:
}
{{- end }}
{{- end }}
}
}
}(namespace)
}
}
}
}
}
}()
}
{{- range .Resources}}
{{- if .ClusterScoped }}
// TODO-JAKE not sure if ther is anything that will need to be done for cluster
// scoped resources. Verify if this is even used in Gloo or not.
/* Setup cluster-wide watch for {{ .Name }} */
var err error
currentSnapshot.{{ upper_camel .PluralName }},err = c.{{ lower_camel .Name }}.List(clients.ListOpts{Ctx: opts.Ctx, Selector: opts.Selector})
Expand Down Expand Up @@ -347,7 +544,7 @@ func (c *{{ lower_camel .GoName }}Emitter) Snapshots(watchNamespaces []string, o
}
currentSnapshot.{{ upper_camel .PluralName }} = {{ lower_camel .Name }}List.Sort()
{{- end }}
{{- end}}
{{- end }}
}
}
}()
Expand Down
Loading