Skip to content

Commit 308d1d3

Browse files
author
Ravi Sankar Penta
committed
Fix filter namespaces in template router
Remove endpoints key format and route key format assumptions in the template plugin and router code. Get the endpoints and route key related information from one place, this will be less prone to errors and easy to modify in the future if needed.
1 parent ca81cd0 commit 308d1d3

File tree

4 files changed

+194
-73
lines changed

4 files changed

+194
-73
lines changed

pkg/router/template/plugin.go

+21-2
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import (
55
"fmt"
66
"path/filepath"
77
"strconv"
8+
"strings"
89
"text/template"
910
"time"
1011

@@ -19,6 +20,10 @@ import (
1920
unidlingapi "github.com/openshift/origin/pkg/unidling/api"
2021
)
2122

23+
const (
24+
endpointsKeySeparator = "/"
25+
)
26+
2227
// TemplatePlugin implements the router.Plugin interface to provide
2328
// a template based, backend-agnostic router.
2429
type TemplatePlugin struct {
@@ -197,14 +202,28 @@ func (p *TemplatePlugin) Commit() error {
197202

198203
// endpointsKey returns the internal router key to use for the given Endpoints.
199204
func endpointsKey(endpoints *kapi.Endpoints) string {
200-
return fmt.Sprintf("%s/%s", endpoints.Namespace, endpoints.Name)
205+
return fmt.Sprintf("%s%s%s", endpoints.Namespace, endpointsKeySeparator, endpoints.Name)
206+
}
207+
208+
func endpointsKeyFromParts(namespace, name string) string {
209+
return fmt.Sprintf("%s%s%s", namespace, endpointsKeySeparator, name)
210+
}
211+
212+
func getPartsFromEndpointsKey(key string) (string, string) {
213+
tokens := strings.SplitN(key, endpointsKeySeparator, 2)
214+
if len(tokens) != 2 {
215+
glog.Errorf("Expected separator %q not found in endpoints key %q", endpointsKeySeparator, key)
216+
}
217+
namespace := tokens[0]
218+
name := tokens[1]
219+
return namespace, name
201220
}
202221

203222
// peerServiceKey may be used by the underlying router when handling endpoints to identify
204223
// endpoints that belong to its peers. THIS MUST FOLLOW THE KEY STRATEGY OF endpointsKey. It
205224
// receives a NamespacedName that is created from the service that is added by the oadm command
206225
func peerEndpointsKey(namespacedName ktypes.NamespacedName) string {
207-
return fmt.Sprintf("%s/%s", namespacedName.Namespace, namespacedName.Name)
226+
return fmt.Sprintf("%s%s%s", namespacedName.Namespace, endpointsKeySeparator, namespacedName.Name)
208227
}
209228

210229
// createRouterEndpoints creates openshift router endpoints based on k8s endpoints

pkg/router/template/plugin_test.go

+27-29
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,7 @@
11
package templaterouter
22

33
import (
4-
"fmt"
54
"reflect"
6-
"strings"
75
"testing"
86
"time"
97

@@ -173,7 +171,7 @@ func (r *TestRouter) DeleteEndpoints(id string) {
173171

174172
// AddRoute adds a ServiceAliasConfig and associated ServiceUnits for the route
175173
func (r *TestRouter) AddRoute(route *routeapi.Route) {
176-
routeKey := r.routeKey(route)
174+
routeKey := getKey(route)
177175

178176
config := ServiceAliasConfig{
179177
Host: route.Spec.Host,
@@ -195,7 +193,7 @@ func (r *TestRouter) numberOfEndpoints(key string) int32 {
195193

196194
// RemoveRoute removes the service alias config for Route
197195
func (r *TestRouter) RemoveRoute(route *routeapi.Route) {
198-
routeKey := r.routeKey(route)
196+
routeKey := getKey(route)
199197
_, ok := r.State[routeKey]
200198
if !ok {
201199
return
@@ -222,25 +220,25 @@ func (r *TestRouter) FilterNamespaces(namespaces sets.String) {
222220
for k := range r.ServiceUnits {
223221
// TODO: the id of a service unit should be defined inside this class, not passed in from the outside
224222
// remove the leak of the abstraction when we refactor this code
225-
ns := strings.SplitN(k, "/", 2)[0]
223+
ns, _ := getPartsFromEndpointsKey(k)
226224
if namespaces.Has(ns) {
227225
continue
228226
}
229227
delete(r.ServiceUnits, k)
230228
}
231229

232230
for k := range r.State {
233-
ns := strings.SplitN(k, "-", 2)[0]
231+
ns, _ := getPartsFromRouteKey(k)
234232
if namespaces.Has(ns) {
235233
continue
236234
}
237235
delete(r.State, k)
238236
}
239237
}
240238

241-
// routeKey create an identifier for the route consisting of host-path
242-
func (r *TestRouter) routeKey(route *routeapi.Route) string {
243-
return route.Spec.Host + "-" + route.Spec.Path
239+
// getKey create an identifier for the route consisting of host-path
240+
func getKey(route *routeapi.Route) string {
241+
return route.Spec.Host + routeKeySeparator + route.Spec.Path
244242
}
245243

246244
func (r *TestRouter) Commit() {
@@ -499,7 +497,7 @@ func TestHandleRoute(t *testing.T) {
499497
},
500498
},
501499
}
502-
serviceUnitKey := fmt.Sprintf("%s/%s", route.Namespace, route.Spec.To.Name)
500+
serviceUnitKey := endpointsKeyFromParts(route.Namespace, route.Spec.To.Name)
503501

504502
plugin.HandleRoute(watch.Added, route)
505503

@@ -508,10 +506,10 @@ func TestHandleRoute(t *testing.T) {
508506
if !ok {
509507
t.Errorf("TestHandleRoute was unable to find the service unit %s after HandleRoute was called", route.Spec.To.Name)
510508
} else {
511-
serviceAliasCfg, ok := router.State[router.routeKey(route)]
509+
serviceAliasCfg, ok := router.State[getKey(route)]
512510

513511
if !ok {
514-
t.Errorf("TestHandleRoute expected route key %s", router.routeKey(route))
512+
t.Errorf("TestHandleRoute expected route key %s", getKey(route))
515513
} else {
516514
if serviceAliasCfg.Host != route.Spec.Host || serviceAliasCfg.Path != route.Spec.Path {
517515
t.Errorf("Expected route did not match service alias config %v : %v", route, serviceAliasCfg)
@@ -541,7 +539,7 @@ func TestHandleRoute(t *testing.T) {
541539
if err := plugin.HandleRoute(watch.Added, duplicateRoute); err == nil {
542540
t.Fatal("unexpected non-error")
543541
}
544-
if _, ok := router.FindServiceUnit("foo/TestService2"); ok {
542+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService2")); ok {
545543
t.Fatalf("unexpected second unit: %#v", router)
546544
}
547545
if r, ok := plugin.RoutesForHost("www.example.com"); !ok || r[0].Name != "test" {
@@ -559,10 +557,10 @@ func TestHandleRoute(t *testing.T) {
559557
if err := plugin.HandleRoute(watch.Deleted, duplicateRoute); err == nil {
560558
t.Fatal("unexpected non-error")
561559
}
562-
if _, ok := router.FindServiceUnit("foo/TestService2"); ok {
560+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService2")); ok {
563561
t.Fatalf("unexpected second unit: %#v", router)
564562
}
565-
if _, ok := router.FindServiceUnit("foo/TestService"); !ok {
563+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); !ok {
566564
t.Fatalf("unexpected first unit: %#v", router)
567565
}
568566
if r, ok := plugin.RoutesForHost("www.example.com"); !ok || r[0].Name != "test" {
@@ -581,7 +579,7 @@ func TestHandleRoute(t *testing.T) {
581579
if err := plugin.HandleRoute(watch.Added, duplicateRoute); err != nil {
582580
t.Fatal("unexpected error")
583581
}
584-
_, ok = router.FindServiceUnit("foo/TestService2")
582+
_, ok = router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService2"))
585583
if !ok {
586584
t.Fatalf("missing second unit: %#v", router)
587585
}
@@ -602,10 +600,10 @@ func TestHandleRoute(t *testing.T) {
602600
if !ok {
603601
t.Errorf("TestHandleRoute was unable to find the service unit %s after HandleRoute was called", route.Spec.To.Name)
604602
} else {
605-
serviceAliasCfg, ok := router.State[router.routeKey(route)]
603+
serviceAliasCfg, ok := router.State[getKey(route)]
606604

607605
if !ok {
608-
t.Errorf("TestHandleRoute expected route key %s", router.routeKey(route))
606+
t.Errorf("TestHandleRoute expected route key %s", getKey(route))
609607
} else {
610608
if serviceAliasCfg.Host != route.Spec.Host || serviceAliasCfg.Path != route.Spec.Path {
611609
t.Errorf("Expected route did not match service alias config %v : %v", route, serviceAliasCfg)
@@ -627,10 +625,10 @@ func TestHandleRoute(t *testing.T) {
627625
if !ok {
628626
t.Errorf("TestHandleRoute was unable to find the service unit %s after HandleRoute was called", route.Spec.To.Name)
629627
} else {
630-
_, ok := router.State[router.routeKey(route)]
628+
_, ok := router.State[getKey(route)]
631629

632630
if ok {
633-
t.Errorf("TestHandleRoute did not expect route key %s", router.routeKey(route))
631+
t.Errorf("TestHandleRoute did not expect route key %s", getKey(route))
634632
}
635633
}
636634
if plugin.HostLen() != 0 {
@@ -668,7 +666,7 @@ func TestHandleRouteExtendedValidation(t *testing.T) {
668666
},
669667
},
670668
}
671-
serviceUnitKey := fmt.Sprintf("%s/%s", route.Namespace, route.Spec.To.Name)
669+
serviceUnitKey := endpointsKeyFromParts(route.Namespace, route.Spec.To.Name)
672670

673671
plugin.HandleRoute(watch.Added, route)
674672

@@ -677,10 +675,10 @@ func TestHandleRouteExtendedValidation(t *testing.T) {
677675
if !ok {
678676
t.Errorf("TestHandleRoute was unable to find the service unit %s after HandleRoute was called", route.Spec.To.Name)
679677
} else {
680-
serviceAliasCfg, ok := router.State[router.routeKey(route)]
678+
serviceAliasCfg, ok := router.State[getKey(route)]
681679

682680
if !ok {
683-
t.Errorf("TestHandleRoute expected route key %s", router.routeKey(route))
681+
t.Errorf("TestHandleRoute expected route key %s", getKey(route))
684682
} else {
685683
if serviceAliasCfg.Host != route.Spec.Host || serviceAliasCfg.Path != route.Spec.Path {
686684
t.Errorf("Expected route did not match service alias config %v : %v", route, serviceAliasCfg)
@@ -1017,7 +1015,7 @@ func TestNamespaceScopingFromEmpty(t *testing.T) {
10171015
// ignores all events for namespace that doesn't match
10181016
for _, s := range []watch.EventType{watch.Added, watch.Modified, watch.Deleted} {
10191017
plugin.HandleRoute(s, route)
1020-
if _, ok := router.FindServiceUnit("foo/TestService"); ok || plugin.HostLen() != 0 {
1018+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); ok || plugin.HostLen() != 0 {
10211019
t.Errorf("unexpected router state %#v", router)
10221020
}
10231021
}
@@ -1026,29 +1024,29 @@ func TestNamespaceScopingFromEmpty(t *testing.T) {
10261024
plugin.HandleNamespaces(sets.NewString("bar"))
10271025
for _, s := range []watch.EventType{watch.Added, watch.Modified, watch.Deleted} {
10281026
plugin.HandleRoute(s, route)
1029-
if _, ok := router.FindServiceUnit("foo/TestService"); ok || plugin.HostLen() != 0 {
1027+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); ok || plugin.HostLen() != 0 {
10301028
t.Errorf("unexpected router state %#v", router)
10311029
}
10321030
}
10331031

10341032
// allow foo
10351033
plugin.HandleNamespaces(sets.NewString("foo", "bar"))
10361034
plugin.HandleRoute(watch.Added, route)
1037-
if _, ok := router.FindServiceUnit("foo/TestService"); !ok || plugin.HostLen() != 1 {
1035+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); !ok || plugin.HostLen() != 1 {
10381036
t.Errorf("unexpected router state %#v", router)
10391037
}
10401038

10411039
// forbid foo, and make sure it's cleared
10421040
plugin.HandleNamespaces(sets.NewString("bar"))
1043-
if _, ok := router.FindServiceUnit("foo/TestService"); ok || plugin.HostLen() != 0 {
1041+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); ok || plugin.HostLen() != 0 {
10441042
t.Errorf("unexpected router state %#v", router)
10451043
}
10461044
plugin.HandleRoute(watch.Modified, route)
1047-
if _, ok := router.FindServiceUnit("foo/TestService"); ok || plugin.HostLen() != 0 {
1045+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); ok || plugin.HostLen() != 0 {
10481046
t.Errorf("unexpected router state %#v", router)
10491047
}
10501048
plugin.HandleRoute(watch.Added, route)
1051-
if _, ok := router.FindServiceUnit("foo/TestService"); ok || plugin.HostLen() != 0 {
1049+
if _, ok := router.FindServiceUnit(endpointsKeyFromParts("foo", "TestService")); ok || plugin.HostLen() != 0 {
10521050
t.Errorf("unexpected router state %#v", router)
10531051
}
10541052
}

0 commit comments

Comments
 (0)