From 66367b419aba200fe8ee03c6f5acde1efd40f29c Mon Sep 17 00:00:00 2001 From: Bowei Du Date: Wed, 3 Jun 2020 21:02:46 -0700 Subject: [PATCH] Add events for major lifecycle events {Add,Remove}Nodes from InstanceGroup SyncIngress, TranslateIngress, IPChanged, GarbageCollection --- pkg/backends/ig_linker_test.go | 5 ++- pkg/backends/integration_test.go | 3 +- pkg/controller/controller.go | 30 ++++++++------- pkg/controller/controller_test.go | 2 +- pkg/events/events.go | 49 +++++++++++++++++++++++++ pkg/events/events_test.go | 34 +++++++++++++++++ pkg/instances/instances.go | 35 +++++++++++++----- pkg/instances/instances_test.go | 3 +- pkg/loadbalancers/forwarding_rules.go | 10 ++++- pkg/loadbalancers/loadbalancers_test.go | 3 +- pkg/loadbalancers/target_proxies.go | 21 ++++++----- pkg/loadbalancers/url_maps.go | 10 ++++- pkg/test/utils.go | 11 +++++- 13 files changed, 173 insertions(+), 43 deletions(-) create mode 100644 pkg/events/events_test.go diff --git a/pkg/backends/ig_linker_test.go b/pkg/backends/ig_linker_test.go index 969a6ed930..3d5e69e80c 100644 --- a/pkg/backends/ig_linker_test.go +++ b/pkg/backends/ig_linker_test.go @@ -27,6 +27,7 @@ import ( "k8s.io/ingress-gce/pkg/annotations" "k8s.io/ingress-gce/pkg/backends/features" "k8s.io/ingress-gce/pkg/instances" + "k8s.io/ingress-gce/pkg/test" "k8s.io/ingress-gce/pkg/utils" "k8s.io/legacy-cloud-providers/gce" ) @@ -47,7 +48,7 @@ func newTestIGLinker(fakeGCE *gce.Cloud, fakeInstancePool instances.NodePool) *i func TestLink(t *testing.T) { fakeIGs := instances.NewFakeInstanceGroups(sets.NewString(), defaultNamer) - fakeNodePool := instances.NewNodePool(fakeIGs, defaultNamer) + fakeNodePool := instances.NewNodePool(fakeIGs, defaultNamer, &test.FakeRecorderSource{}) fakeGCE := gce.NewFakeGCECloud(gce.DefaultTestClusterValues()) linker := newTestIGLinker(fakeGCE, fakeNodePool) @@ -77,7 +78,7 @@ func TestLink(t *testing.T) { func TestLinkWithCreationModeError(t *testing.T) { fakeIGs := instances.NewFakeInstanceGroups(sets.NewString(), defaultNamer) - fakeNodePool := instances.NewNodePool(fakeIGs, defaultNamer) + fakeNodePool := instances.NewNodePool(fakeIGs, defaultNamer, &test.FakeRecorderSource{}) fakeGCE := gce.NewFakeGCECloud(gce.DefaultTestClusterValues()) linker := newTestIGLinker(fakeGCE, fakeNodePool) diff --git a/pkg/backends/integration_test.go b/pkg/backends/integration_test.go index e3d168b620..8389cf5e89 100644 --- a/pkg/backends/integration_test.go +++ b/pkg/backends/integration_test.go @@ -29,6 +29,7 @@ import ( "k8s.io/ingress-gce/pkg/annotations" "k8s.io/ingress-gce/pkg/healthchecks" "k8s.io/ingress-gce/pkg/instances" + "k8s.io/ingress-gce/pkg/test" "k8s.io/ingress-gce/pkg/utils" "k8s.io/legacy-cloud-providers/gce" ) @@ -45,7 +46,7 @@ func newTestJig(fakeGCE *gce.Cloud) *Jig { fakeBackendPool := NewPool(fakeGCE, defaultNamer) fakeIGs := instances.NewFakeInstanceGroups(sets.NewString(), defaultNamer) - fakeInstancePool := instances.NewNodePool(fakeIGs, defaultNamer) + fakeInstancePool := instances.NewNodePool(fakeIGs, defaultNamer, &test.FakeRecorderSource{}) fakeInstancePool.Init(&instances.FakeZoneLister{Zones: []string{defaultZone}}) // Add standard hooks for mocking update calls. Each test can set a different update hook if it chooses to. diff --git a/pkg/controller/controller.go b/pkg/controller/controller.go index 4fb268b8de..cd758093ed 100644 --- a/pkg/controller/controller.go +++ b/pkg/controller/controller.go @@ -41,6 +41,7 @@ import ( "k8s.io/ingress-gce/pkg/common/operator" "k8s.io/ingress-gce/pkg/context" "k8s.io/ingress-gce/pkg/controller/translator" + "k8s.io/ingress-gce/pkg/events" "k8s.io/ingress-gce/pkg/flags" "k8s.io/ingress-gce/pkg/frontendconfig" "k8s.io/ingress-gce/pkg/healthchecks" @@ -108,7 +109,7 @@ func NewLoadBalancerController( }) healthChecker := healthchecks.NewHealthChecker(ctx.Cloud, ctx.HealthCheckPath, ctx.DefaultBackendSvcPort.ID.Service) - instancePool := instances.NewNodePool(ctx.Cloud, ctx.ClusterNamer) + instancePool := instances.NewNodePool(ctx.Cloud, ctx.ClusterNamer, ctx) backendPool := backends.NewPool(ctx.Cloud, ctx.ClusterNamer) lbc := LoadBalancerController{ @@ -139,8 +140,8 @@ func NewLoadBalancerController( return } - klog.V(3).Infof("Ingress %v added, enqueuing", common.NamespacedName(addIng)) - lbc.ctx.Recorder(addIng.Namespace).Eventf(addIng, apiv1.EventTypeNormal, "ADD", common.NamespacedName(addIng)) + klog.V(2).Infof("Ingress %v added, enqueuing", common.NamespacedName(addIng)) + lbc.ctx.Recorder(addIng.Namespace).Eventf(addIng, apiv1.EventTypeNormal, events.SyncIngress, "Scheduled for sync") lbc.ingQueue.Enqueue(obj) }, DeleteFunc: func(obj interface{}) { @@ -176,11 +177,11 @@ func NewLoadBalancerController( return } if reflect.DeepEqual(old, cur) { - klog.V(3).Infof("Periodic enqueueing of %v", common.NamespacedName(curIng)) + klog.V(2).Infof("Periodic enqueueing of %v", common.NamespacedName(curIng)) } else { - klog.V(3).Infof("Ingress %v changed, enqueuing", common.NamespacedName(curIng)) + klog.V(2).Infof("Ingress %v changed, enqueuing", common.NamespacedName(curIng)) } - + lbc.ctx.Recorder(curIng.Namespace).Eventf(curIng, apiv1.EventTypeNormal, events.SyncIngress, "Scheduled for sync") lbc.ingQueue.Enqueue(cur) }, }) @@ -558,7 +559,8 @@ func (lbc *LoadBalancerController) sync(key string) error { err := lbc.ingSyncer.GC(allIngresses, ing, frontendGCAlgorithm) // Skip emitting an event if ingress does not exist as we cannot retrieve ingress namespace. if err != nil && ingExists { - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "GC", fmt.Sprintf("Error during GC: %v", err)) + klog.Errorf("Error in GC for %s/%s: %v", ing.Namespace, ing.Name, err) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.GarbageCollection, "Error: %v", err) } // Delete the ingress state for metrics after GC is successful. if err == nil && ingExists { @@ -578,8 +580,8 @@ func (lbc *LoadBalancerController) sync(key string) error { urlMap, errs := lbc.Translator.TranslateIngress(ing, lbc.ctx.DefaultBackendSvcPort.ID, lbc.ctx.ClusterNamer) if errs != nil { - msg := fmt.Errorf("error while evaluating the ingress spec: %v", utils.JoinErrs(errs)) - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "Translate", msg.Error()) + msg := fmt.Errorf("invalid ingress spec: %v", utils.JoinErrs(errs)) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.TranslateIngress, "Translation failed: %v", msg) return msg } @@ -587,7 +589,7 @@ func (lbc *LoadBalancerController) sync(key string) error { syncState := &syncState{urlMap, ing, nil} syncErr := lbc.ingSyncer.Sync(syncState) if syncErr != nil { - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "Sync", fmt.Sprintf("Error during sync: %v", syncErr.Error())) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.SyncIngress, "Error syncing to GCP: %v", syncErr.Error()) } else { // Insert/update the ingress state for metrics after successful sync. lbc.metrics.SetIngress(key, metrics.NewIngressState(ing, urlMap.AllServicePorts())) @@ -598,7 +600,7 @@ func (lbc *LoadBalancerController) sync(key string) error { // free up enough quota for the next sync to pass. frontendGCAlgorithm := frontendGCAlgorithm(ingExists, ing) if gcErr := lbc.ingSyncer.GC(allIngresses, ing, frontendGCAlgorithm); gcErr != nil { - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "GC", fmt.Sprintf("Error during GC: %v", gcErr)) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.GarbageCollection, "Error during garbage collection: %v", gcErr) return fmt.Errorf("error during sync %v, error during GC %v", syncErr, gcErr) } @@ -633,7 +635,7 @@ func (lbc *LoadBalancerController) updateIngressStatus(l7 *loadbalancers.L7, ing klog.Errorf("PatchIngressStatus(%s/%s) failed: %v", currIng.Namespace, currIng.Name, err) return err } - lbc.ctx.Recorder(ing.Namespace).Eventf(currIng, apiv1.EventTypeNormal, "CREATE", "ip: %v", ip) + lbc.ctx.Recorder(ing.Namespace).Eventf(currIng, apiv1.EventTypeNormal, events.IPChanged, "IP is now %v", ip) } } annotations, err := loadbalancers.GetLBAnnotations(l7, currIng.Annotations, lbc.backendSyncer) @@ -655,7 +657,7 @@ func (lbc *LoadBalancerController) toRuntimeInfo(ing *v1beta1.Ingress, urlMap *u if apierrors.IsNotFound(err) { // TODO: this path should be removed when external certificate managers migrate to a better solution. const msg = "Could not find TLS certificates. Continuing setup for the load balancer to serve HTTP. Note: this behavior is deprecated and will be removed in a future version of ingress-gce" - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "Sync", msg) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.SyncIngress, msg) } else { klog.Errorf("Could not get certificates for ingress %s/%s: %v", ing.Namespace, ing.Name, err) return nil, err @@ -666,7 +668,7 @@ func (lbc *LoadBalancerController) toRuntimeInfo(ing *v1beta1.Ingress, urlMap *u if lbc.ctx.FrontendConfigEnabled { feConfig, err = frontendconfig.FrontendConfigForIngress(lbc.ctx.FrontendConfigs().List(), ing) if err != nil { - lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, "Sync", fmt.Sprintf("%v", err)) + lbc.ctx.Recorder(ing.Namespace).Eventf(ing, apiv1.EventTypeWarning, events.SyncIngress, "Error: %v", err) } // Object in cache could be changed in-flight. Deepcopy to // reduce race conditions. diff --git a/pkg/controller/controller_test.go b/pkg/controller/controller_test.go index caf9d07c83..a5e0cd284d 100644 --- a/pkg/controller/controller_test.go +++ b/pkg/controller/controller_test.go @@ -77,7 +77,7 @@ func newLoadBalancerController() *LoadBalancerController { ctx := context.NewControllerContext(nil, kubeClient, backendConfigClient, nil, fakeGCE, namer, "" /*kubeSystemUID*/, ctxConfig) lbc := NewLoadBalancerController(ctx, stopCh) // TODO(rramkumar): Fix this so we don't have to override with our fake - lbc.instancePool = instances.NewNodePool(instances.NewFakeInstanceGroups(sets.NewString(), namer), namer) + lbc.instancePool = instances.NewNodePool(instances.NewFakeInstanceGroups(sets.NewString(), namer), namer, &test.FakeRecorderSource{}) lbc.l7Pool = loadbalancers.NewLoadBalancerPool(fakeGCE, namer, events.RecorderProducerMock{}, namer_util.NewFrontendNamerFactory(namer, "")) lbc.instancePool.Init(&instances.FakeZoneLister{Zones: []string{fakeZone}}) diff --git a/pkg/events/events.go b/pkg/events/events.go index bb291bc32f..ad6d55ca9c 100644 --- a/pkg/events/events.go +++ b/pkg/events/events.go @@ -17,9 +17,22 @@ limitations under the License. package events import ( + v1 "k8s.io/api/core/v1" "k8s.io/client-go/tools/record" ) +const ( + AddNodes = "IngressGCE_AddNodes" + RemoveNodes = "IngressGCE_RemoveNodes" + + SyncIngress = "Sync" + TranslateIngress = "Translate" + IPChanged = "IPChanged" + GarbageCollection = "GarbageCollection" + + SyncService = "Sync" +) + type RecorderProducer interface { Recorder(ns string) record.EventRecorder } @@ -30,3 +43,39 @@ type RecorderProducerMock struct { func (r RecorderProducerMock) Recorder(ns string) record.EventRecorder { return &record.FakeRecorder{} } + +// GloablEventf records a Cluster level event not attached to a given object. +func GlobalEventf(r record.EventRecorder, eventtype, reason, messageFmt string, args ...interface{}) { + // Using an empty ObjectReference to indicate no associated + // resource. This apparently works, see the package + // k8s.io/client-go/tools/record. + r.Eventf(&v1.ObjectReference{}, eventtype, reason, messageFmt, args...) +} + +// truncatedStringListMax is a variable to make testing easier. This +// value should not be modified. +var truncatedStringListMax = 2000 + +// TruncateStringList will render the list of items as a string, +// eliding elements with elipsis at the end if there are more than a +// reasonable number of characters in the resulting string. This is +// used to prevent accidentally dumping enormous strings into the +// Event description. +func TruncatedStringList(items []string) string { + var ( + ret = "[" + first = true + ) + for _, s := range items { + if len(ret)+len(s)+1 > truncatedStringListMax { + ret += ", ..." + break + } + if !first { + ret += ", " + } + first = false + ret += s + } + return ret + "]" +} diff --git a/pkg/events/events_test.go b/pkg/events/events_test.go new file mode 100644 index 0000000000..14699336ef --- /dev/null +++ b/pkg/events/events_test.go @@ -0,0 +1,34 @@ +package events + +import ( + "fmt" + "testing" +) + +func TestTruncatedStringList(t *testing.T) { + var saved int + truncatedStringListMax, saved = 30, truncatedStringListMax + defer func() { truncatedStringListMax = saved }() + + for _, tc := range []struct { + desc string + count int + want string + }{ + {"zero", 0, "[]"}, + {"one", 1, "[elt-0]"}, + {"not truncated", 4, "[elt-0, elt-1, elt-2, elt-3]"}, + {"truncated", 20, "[elt-0, elt-1, elt-2, elt-3, ...]"}, + } { + t.Run(tc.desc, func(t *testing.T) { + var l []string + for i := 0; i < tc.count; i++ { + l = append(l, fmt.Sprintf("elt-%d", i)) + } + got := TruncatedStringList(l) + if got != tc.want { + t.Errorf("TruncatedString(%v) = %q; want %q", l, got, tc.want) + } + }) + } +} diff --git a/pkg/instances/instances.go b/pkg/instances/instances.go index 324aba0ec1..fb8a5b03eb 100644 --- a/pkg/instances/instances.go +++ b/pkg/instances/instances.go @@ -21,10 +21,13 @@ import ( "net/http" "time" + "google.golang.org/api/compute/v1" + "k8s.io/client-go/tools/record" + "k8s.io/ingress-gce/pkg/events" "k8s.io/ingress-gce/pkg/utils/namer" "k8s.io/klog" - "google.golang.org/api/compute/v1" + core "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/util/sets" "k8s.io/ingress-gce/pkg/utils" @@ -39,16 +42,22 @@ const ( type Instances struct { cloud InstanceGroups ZoneLister - namer namer.BackendNamer + namer namer.BackendNamer + recorder record.EventRecorder +} + +type recorderSource interface { + Recorder(ns string) record.EventRecorder } // NewNodePool creates a new node pool. // - cloud: implements InstanceGroups, used to sync Kubernetes nodes with // members of the cloud InstanceGroup. -func NewNodePool(cloud InstanceGroups, namer namer.BackendNamer) NodePool { +func NewNodePool(cloud InstanceGroups, namer namer.BackendNamer, recorders recorderSource) NodePool { return &Instances{ - cloud: cloud, - namer: namer, + cloud: cloud, + namer: namer, + recorder: recorders.Recorder(""), // No namespace } } @@ -250,7 +259,8 @@ func (i *Instances) splitNodesByZone(names []string) map[string][]string { // Add adds the given instances to the appropriately zoned Instance Group. func (i *Instances) Add(groupName string, names []string) error { - errs := []error{} + events.GlobalEventf(i.recorder, core.EventTypeNormal, events.AddNodes, "Adding %s to InstanceGroup %q", events.TruncatedStringList(names), groupName) + var errs []error for zone, nodeNames := range i.splitNodesByZone(names) { klog.V(1).Infof("Adding nodes %v to %v in zone %v", nodeNames, groupName, zone) if err := i.cloud.AddInstancesToInstanceGroup(groupName, zone, i.cloud.ToInstanceReferences(zone, nodeNames)); err != nil { @@ -260,12 +270,16 @@ func (i *Instances) Add(groupName string, names []string) error { if len(errs) == 0 { return nil } - return fmt.Errorf("%v", errs) + + err := fmt.Errorf("AddInstances: %v", errs) + events.GlobalEventf(i.recorder, core.EventTypeWarning, events.AddNodes, "Error adding %s to InstanceGroup %q: %v", events.TruncatedStringList(names), groupName, err) + return err } // Remove removes the given instances from the appropriately zoned Instance Group. func (i *Instances) Remove(groupName string, names []string) error { - errs := []error{} + events.GlobalEventf(i.recorder, core.EventTypeNormal, events.RemoveNodes, "Removing %s from InstanceGroup %q", events.TruncatedStringList(names), groupName) + var errs []error for zone, nodeNames := range i.splitNodesByZone(names) { klog.V(1).Infof("Removing nodes %v from %v in zone %v", nodeNames, groupName, zone) if err := i.cloud.RemoveInstancesFromInstanceGroup(groupName, zone, i.cloud.ToInstanceReferences(zone, nodeNames)); err != nil { @@ -275,7 +289,10 @@ func (i *Instances) Remove(groupName string, names []string) error { if len(errs) == 0 { return nil } - return fmt.Errorf("%v", errs) + + err := fmt.Errorf("RemoveInstances: %v", errs) + events.GlobalEventf(i.recorder, core.EventTypeWarning, events.RemoveNodes, "Error removing nodes %s from InstanceGroup %q: %v", events.TruncatedStringList(names), groupName, err) + return err } // Sync nodes with the instances in the instance group. diff --git a/pkg/instances/instances_test.go b/pkg/instances/instances_test.go index 170649fcde..53acd7eef4 100644 --- a/pkg/instances/instances_test.go +++ b/pkg/instances/instances_test.go @@ -20,6 +20,7 @@ import ( "testing" "k8s.io/apimachinery/pkg/util/sets" + "k8s.io/ingress-gce/pkg/test" "k8s.io/ingress-gce/pkg/utils/namer" ) @@ -28,7 +29,7 @@ const defaultZone = "default-zone" var defaultNamer = namer.NewNamer("uid1", "fw1") func newNodePool(f *FakeInstanceGroups, zone string) NodePool { - pool := NewNodePool(f, defaultNamer) + pool := NewNodePool(f, defaultNamer, &test.FakeRecorderSource{}) pool.Init(&FakeZoneLister{[]string{zone}}) return pool } diff --git a/pkg/loadbalancers/forwarding_rules.go b/pkg/loadbalancers/forwarding_rules.go index 69aa9a4844..a70ca157e0 100644 --- a/pkg/loadbalancers/forwarding_rules.go +++ b/pkg/loadbalancers/forwarding_rules.go @@ -21,10 +21,12 @@ import ( "github.com/GoogleCloudPlatform/k8s-cloud-provider/pkg/cloud" "github.com/GoogleCloudPlatform/k8s-cloud-provider/pkg/cloud/meta" + corev1 "k8s.io/api/core/v1" v1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/types" "k8s.io/ingress-gce/pkg/annotations" "k8s.io/ingress-gce/pkg/composite" + "k8s.io/ingress-gce/pkg/events" "k8s.io/ingress-gce/pkg/flags" "k8s.io/ingress-gce/pkg/translator" "k8s.io/ingress-gce/pkg/utils" @@ -92,6 +94,7 @@ func (l *L7) checkForwardingRule(protocol namer.NamerProtocol, name, proxyLink, return nil, err } existing = nil + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "ForwardingRule %q deleted", key.Name) } if existing == nil { // This is a special case where exactly one of http or https forwarding rule @@ -115,6 +118,8 @@ func (l *L7) checkForwardingRule(protocol namer.NamerProtocol, name, proxyLink, if err = composite.CreateForwardingRule(l.cloud, key, fr); err != nil { return nil, err } + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "ForwardingRule %q created", key.Name) + key, err = l.CreateKey(name) if err != nil { return nil, err @@ -293,6 +298,7 @@ func (l *L4) ensureForwardingRule(loadBalancerName, bsLink string, options gce.I if err = utils.IgnoreHTTPNotFound(composite.DeleteForwardingRule(l.cloud, key, existingVersion)); err != nil { return nil, err } + l.recorder.Eventf(l.Service, corev1.EventTypeNormal, events.SyncIngress, "ForwardingRule %q deleted", key.Name) } } klog.V(2).Infof("ensureForwardingRule: Recreating forwarding rule - %s", fr.Name) @@ -302,9 +308,11 @@ func (l *L4) ensureForwardingRule(loadBalancerName, bsLink string, options gce.I if addrMgr != nil { // Now that the controller knows the forwarding rule exists, we can release the address. if err := addrMgr.ReleaseAddress(); err != nil { - klog.Errorf("ensureInternalLoadBalancer: failed to release address reservation, possibly causing an orphan: %v", err) + klog.Errorf("ensureInternalLoadBalancer: - %s, failed to release address reservation, possibly causing an orphan: %v", fr.Name, err) } } + l.recorder.Eventf(l.Service, corev1.EventTypeNormal, events.SyncIngress, "ForwardingRule %q created", key.Name) + return composite.GetForwardingRule(l.cloud, key, fr.Version) } diff --git a/pkg/loadbalancers/loadbalancers_test.go b/pkg/loadbalancers/loadbalancers_test.go index 7898b66796..44404e88be 100644 --- a/pkg/loadbalancers/loadbalancers_test.go +++ b/pkg/loadbalancers/loadbalancers_test.go @@ -40,6 +40,7 @@ import ( "k8s.io/ingress-gce/pkg/flags" "k8s.io/ingress-gce/pkg/instances" "k8s.io/ingress-gce/pkg/loadbalancers/features" + "k8s.io/ingress-gce/pkg/test" "k8s.io/ingress-gce/pkg/translator" "k8s.io/ingress-gce/pkg/utils" "k8s.io/ingress-gce/pkg/utils/common" @@ -143,7 +144,7 @@ func newIngress() *v1beta1.Ingress { func newFakeLoadBalancerPool(cloud *gce.Cloud, t *testing.T, namer *namer_util.Namer) L7s { fakeIGs := instances.NewFakeInstanceGroups(sets.NewString(), namer) - nodePool := instances.NewNodePool(fakeIGs, namer) + nodePool := instances.NewNodePool(fakeIGs, namer, &test.FakeRecorderSource{}) nodePool.Init(&instances.FakeZoneLister{Zones: []string{defaultZone}}) return L7s{cloud, namer, events.RecorderProducerMock{}, namer_util.NewFrontendNamerFactory(namer, "")} diff --git a/pkg/loadbalancers/target_proxies.go b/pkg/loadbalancers/target_proxies.go index 5de9276fb3..b9c0f690fa 100644 --- a/pkg/loadbalancers/target_proxies.go +++ b/pkg/loadbalancers/target_proxies.go @@ -17,7 +17,9 @@ limitations under the License. package loadbalancers import ( + corev1 "k8s.io/api/core/v1" "k8s.io/ingress-gce/pkg/composite" + "k8s.io/ingress-gce/pkg/events" "k8s.io/ingress-gce/pkg/flags" "k8s.io/ingress-gce/pkg/translator" "k8s.io/ingress-gce/pkg/utils" @@ -55,7 +57,6 @@ func (l *L7) checkProxy() (err error) { currentProxy, _ := composite.GetTargetHttpProxy(l.cloud, key, version) if currentProxy == nil { klog.V(3).Infof("Creating new http proxy for urlmap %v", l.um.Name) - key, err := l.CreateKey(proxy.Name) if err != nil { return err @@ -63,11 +64,8 @@ func (l *L7) checkProxy() (err error) { if err = composite.CreateTargetHttpProxy(l.cloud, key, proxy); err != nil { return err } - key, err = l.CreateKey(proxy.Name) - if err != nil { - return err - } currentProxy, err = composite.GetTargetHttpProxy(l.cloud, key, version) + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q created", key.Name) if err != nil { return err } @@ -84,6 +82,7 @@ func (l *L7) checkProxy() (err error) { if err := composite.SetUrlMapForTargetHttpProxy(l.cloud, key, currentProxy, proxy.UrlMap); err != nil { return err } + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q updated", key.Name) } l.tp = currentProxy return nil @@ -95,7 +94,7 @@ func (l *L7) checkHttpsProxy() (err error) { env := &translator.Env{FrontendConfig: l.runtimeInfo.FrontendConfig} if len(l.sslCerts) == 0 { - klog.V(3).Infof("No SSL certificates for %q, will not create HTTPS Proxy.", l) + klog.V(2).Infof("No SSL certificates for %q, will not create HTTPS Proxy.", l) return nil } @@ -126,6 +125,7 @@ func (l *L7) checkHttpsProxy() (err error) { if err = composite.CreateTargetHttpsProxy(l.cloud, key, proxy); err != nil { return err } + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q created", key.Name) key, err = l.CreateKey(proxy.Name) if err != nil { @@ -139,9 +139,9 @@ func (l *L7) checkHttpsProxy() (err error) { l.tps = currentProxy return nil } + if !utils.EqualResourcePaths(currentProxy.UrlMap, proxy.UrlMap) { - klog.V(3).Infof("Https Proxy %v has the wrong url map, setting %v overwriting %v", - currentProxy.Name, proxy.UrlMap, currentProxy.UrlMap) + klog.V(2).Infof("Https Proxy %v has the wrong url map, setting %v overwriting %v", currentProxy.Name, proxy.UrlMap, currentProxy.UrlMap) key, err := l.CreateKey(currentProxy.Name) if err != nil { return err @@ -149,10 +149,11 @@ func (l *L7) checkHttpsProxy() (err error) { if err := composite.SetUrlMapForTargetHttpsProxy(l.cloud, key, currentProxy, proxy.UrlMap); err != nil { return err } + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q updated", key.Name) } if !l.compareCerts(currentProxy.SslCertificates) { - klog.V(3).Infof("Https Proxy %q has the wrong ssl certs, setting %v overwriting %v", + klog.V(2).Infof("Https Proxy %q has the wrong ssl certs, setting %v overwriting %v", currentProxy.Name, toCertNames(l.sslCerts), currentProxy.SslCertificates) var sslCertURLs []string for _, cert := range l.sslCerts { @@ -165,6 +166,7 @@ func (l *L7) checkHttpsProxy() (err error) { if err := composite.SetSslCertificateForTargetHttpsProxy(l.cloud, key, currentProxy, sslCertURLs); err != nil { return err } + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q certs updated", key.Name) } if flags.F.EnableFrontendConfig && sslPolicySet { @@ -200,6 +202,7 @@ func (l *L7) ensureSslPolicy(env *translator.Env, currentProxy *composite.Target return err } if err := composite.SetSslPolicyForTargetHttpsProxy(l.cloud, key, currentProxy, policyLink); err != nil { + l.recorder.Eventf(l.runtimeInfo.Ingress, corev1.EventTypeNormal, events.SyncIngress, "TargetProxy %q SSLPolicy updated", key.Name) return err } } diff --git a/pkg/loadbalancers/url_maps.go b/pkg/loadbalancers/url_maps.go index d673c1cabe..3c174d064b 100644 --- a/pkg/loadbalancers/url_maps.go +++ b/pkg/loadbalancers/url_maps.go @@ -19,8 +19,10 @@ package loadbalancers import ( "fmt" + apiv1 "k8s.io/api/core/v1" "k8s.io/apimachinery/pkg/util/sets" "k8s.io/ingress-gce/pkg/composite" + "k8s.io/ingress-gce/pkg/events" "k8s.io/ingress-gce/pkg/translator" "k8s.io/ingress-gce/pkg/utils" "k8s.io/klog" @@ -50,11 +52,13 @@ func (l *L7) ensureComputeURLMap() error { } if currentMap == nil { - klog.V(3).Infof("Creating URLMap %q", expectedMap.Name) + klog.V(2).Infof("Creating URLMap %q", expectedMap.Name) if err := composite.CreateUrlMap(l.cloud, key, expectedMap); err != nil { return fmt.Errorf("CreateUrlMap: %v", err) } + l.recorder.Eventf(&l.ingress, apiv1.EventTypeNormal, events.SyncIngress, "UrlMap %q created", key.Name) l.um = expectedMap + return nil } @@ -64,13 +68,15 @@ func (l *L7) ensureComputeURLMap() error { return nil } - klog.V(3).Infof("Updating URLMap for %q", l) + klog.V(2).Infof("Updating URLMap for %q", l) expectedMap.Fingerprint = currentMap.Fingerprint if err := composite.UpdateUrlMap(l.cloud, key, expectedMap); err != nil { return fmt.Errorf("UpdateURLMap: %v", err) } + l.recorder.Eventf(&l.ingress, apiv1.EventTypeNormal, events.SyncIngress, "UrlMap %q updated", key.Name) l.um = expectedMap + return nil } diff --git a/pkg/test/utils.go b/pkg/test/utils.go index 087593b47b..e54124a36d 100644 --- a/pkg/test/utils.go +++ b/pkg/test/utils.go @@ -2,6 +2,9 @@ package test import ( "fmt" + "strings" + "time" + "github.com/GoogleCloudPlatform/k8s-cloud-provider/pkg/cloud" "github.com/GoogleCloudPlatform/k8s-cloud-provider/pkg/cloud/meta" "google.golang.org/api/compute/v1" @@ -16,8 +19,6 @@ import ( backendconfig "k8s.io/ingress-gce/pkg/apis/backendconfig/v1" "k8s.io/ingress-gce/pkg/utils" "k8s.io/legacy-cloud-providers/gce" - "strings" - "time" ) const ( @@ -246,3 +247,9 @@ func Float64ToPtr(val float64) *float64 { func Int64ToPtr(val int64) *int64 { return &val } + +type FakeRecorderSource struct{} + +func (_ *FakeRecorderSource) Recorder(ns string) record.EventRecorder { + return record.NewFakeRecorder(100) +}