mirror of
https://github.com/status-im/consul.git
synced 2025-01-28 14:34:59 +00:00
11d6b0df45
xRoute resource types contain a slice of parentRefs to services that they manipulate traffic for. All xRoutes that have a parentRef to given Service will be merged together to generate a ComputedRoutes resource name-aligned with that Service. This means that a write of an xRoute with 2 parent ref pointers will cause at most 2 reconciles for ComputedRoutes. If that xRoute's list of parentRefs were ever to be reduced, or otherwise lose an item, that subsequent map event will only emit events for the current set of refs. The removed ref will not cause the generated ComputedRoutes related to that service to be re-reconciled to omit the influence of that xRoute. To combat this, we will store on the ComputedRoutes resource a BoundResources []*pbresource.Reference field with references to all resources that were used to influence the generated output. When the routes controller reconciles, it will use a bimapper to index this influence, and the dependency mappers for the xRoutes will look themselves up in that index to discover additional (former) ComputedRoutes that need to be notified as well.
490 lines
15 KiB
Go
490 lines
15 KiB
Go
// Copyright (c) HashiCorp, Inc.
|
|
// SPDX-License-Identifier: BUSL-1.1
|
|
|
|
package bimapper
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/hashicorp/consul/internal/controller"
|
|
"github.com/hashicorp/consul/internal/resource"
|
|
rtest "github.com/hashicorp/consul/internal/resource/resourcetest"
|
|
"github.com/hashicorp/consul/proto-public/pbresource"
|
|
"github.com/hashicorp/consul/proto/private/prototest"
|
|
)
|
|
|
|
const (
|
|
fakeGroupName = "catalog"
|
|
fakeVersion = "v1"
|
|
)
|
|
|
|
var (
|
|
fakeFooType = &pbresource.Type{
|
|
Group: fakeGroupName,
|
|
GroupVersion: fakeVersion,
|
|
Kind: "Foo",
|
|
}
|
|
fakeBarType = &pbresource.Type{
|
|
Group: fakeGroupName,
|
|
GroupVersion: fakeVersion,
|
|
Kind: "Bar",
|
|
}
|
|
fakeBazType = &pbresource.Type{
|
|
Group: fakeGroupName,
|
|
GroupVersion: fakeVersion,
|
|
Kind: "Baz",
|
|
}
|
|
)
|
|
|
|
func TestMapper(t *testing.T) {
|
|
// Create an advance pointer to some services.
|
|
|
|
randoSvc := rtest.Resource(fakeBarType, "rando").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
apiSvc := rtest.Resource(fakeBarType, "api").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
fooSvc := rtest.Resource(fakeBarType, "foo").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
barSvc := rtest.Resource(fakeBarType, "bar").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
wwwSvc := rtest.Resource(fakeBarType, "www").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
|
|
apiRef := newRef(fakeBarType, "api")
|
|
fooRef := newRef(fakeBarType, "foo")
|
|
barRef := newRef(fakeBarType, "bar")
|
|
wwwRef := newRef(fakeBarType, "www")
|
|
|
|
fail1 := rtest.Resource(fakeFooType, "api").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
fail1Refs := []resource.ReferenceOrID{
|
|
apiRef,
|
|
fooRef,
|
|
barRef,
|
|
}
|
|
|
|
fail2 := rtest.Resource(fakeFooType, "www").WithTenancy(resource.DefaultNamespacedTenancy()).Build()
|
|
fail2Refs := []resource.ReferenceOrID{
|
|
wwwRef,
|
|
fooRef,
|
|
}
|
|
|
|
fail1UpdatedRefs := []resource.ReferenceOrID{
|
|
apiRef,
|
|
barRef,
|
|
}
|
|
|
|
m := New(fakeFooType, fakeBarType)
|
|
|
|
// Nothing tracked yet so we assume nothing.
|
|
requireLinksForItem(t, m, fail1.Id)
|
|
requireLinksForItem(t, m, fail2.Id)
|
|
requireItemsForLink(t, m, apiRef)
|
|
requireItemsForLink(t, m, fooRef)
|
|
requireItemsForLink(t, m, barRef)
|
|
requireItemsForLink(t, m, wwwRef)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc)
|
|
requireServicesTracked(t, m, fooSvc)
|
|
requireServicesTracked(t, m, barSvc)
|
|
requireServicesTracked(t, m, wwwSvc)
|
|
|
|
// no-ops
|
|
m.UntrackItem(fail1.Id)
|
|
|
|
// still nothing
|
|
requireLinksForItem(t, m, fail1.Id)
|
|
requireLinksForItem(t, m, fail2.Id)
|
|
requireItemsForLink(t, m, apiRef)
|
|
requireItemsForLink(t, m, fooRef)
|
|
requireItemsForLink(t, m, barRef)
|
|
requireItemsForLink(t, m, wwwRef)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc)
|
|
requireServicesTracked(t, m, fooSvc)
|
|
requireServicesTracked(t, m, barSvc)
|
|
requireServicesTracked(t, m, wwwSvc)
|
|
|
|
// Actually insert some data.
|
|
m.TrackItem(fail1.Id, fail1Refs)
|
|
|
|
// Check links mapping
|
|
requireLinksForItem(t, m, fail1.Id, fail1Refs...)
|
|
|
|
requireLinksForItem(t, m, fail1.Id, fail1Refs...)
|
|
requireItemsForLink(t, m, apiRef, fail1.Id)
|
|
requireItemsForLink(t, m, fooRef, fail1.Id)
|
|
requireItemsForLink(t, m, barRef, fail1.Id)
|
|
requireItemsForLink(t, m, wwwRef)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc, fail1.Id)
|
|
requireServicesTracked(t, m, fooSvc, fail1.Id)
|
|
requireServicesTracked(t, m, barSvc, fail1.Id)
|
|
requireServicesTracked(t, m, wwwSvc)
|
|
|
|
// track it again, no change
|
|
m.TrackItem(fail1.Id, fail1Refs)
|
|
|
|
requireLinksForItem(t, m, fail1.Id, fail1Refs...)
|
|
requireItemsForLink(t, m, apiRef, fail1.Id)
|
|
requireItemsForLink(t, m, fooRef, fail1.Id)
|
|
requireItemsForLink(t, m, barRef, fail1.Id)
|
|
requireItemsForLink(t, m, wwwRef)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc, fail1.Id)
|
|
requireServicesTracked(t, m, fooSvc, fail1.Id)
|
|
requireServicesTracked(t, m, barSvc, fail1.Id)
|
|
requireServicesTracked(t, m, wwwSvc)
|
|
|
|
// track new one that overlaps slightly
|
|
m.TrackItem(fail2.Id, fail2Refs)
|
|
|
|
// Check links mapping for the new one
|
|
requireLinksForItem(t, m, fail1.Id, fail1Refs...)
|
|
requireLinksForItem(t, m, fail2.Id, fail2Refs...)
|
|
requireItemsForLink(t, m, apiRef, fail1.Id)
|
|
requireItemsForLink(t, m, fooRef, fail1.Id, fail2.Id)
|
|
requireItemsForLink(t, m, barRef, fail1.Id)
|
|
requireItemsForLink(t, m, wwwRef, fail2.Id)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc, fail1.Id)
|
|
requireServicesTracked(t, m, fooSvc, fail1.Id, fail2.Id)
|
|
requireServicesTracked(t, m, barSvc, fail1.Id)
|
|
requireServicesTracked(t, m, wwwSvc, fail2.Id)
|
|
|
|
// update the original to change it
|
|
m.TrackItem(fail1.Id, fail1UpdatedRefs)
|
|
|
|
requireLinksForItem(t, m, fail1.Id, fail1UpdatedRefs...)
|
|
requireLinksForItem(t, m, fail2.Id, fail2Refs...)
|
|
requireItemsForLink(t, m, apiRef, fail1.Id)
|
|
requireItemsForLink(t, m, fooRef, fail2.Id)
|
|
requireItemsForLink(t, m, barRef, fail1.Id)
|
|
requireItemsForLink(t, m, wwwRef, fail2.Id)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc, fail1.Id)
|
|
requireServicesTracked(t, m, fooSvc, fail2.Id)
|
|
requireServicesTracked(t, m, barSvc, fail1.Id)
|
|
requireServicesTracked(t, m, wwwSvc, fail2.Id)
|
|
|
|
// delete the original
|
|
m.UntrackItem(fail1.Id)
|
|
|
|
requireLinksForItem(t, m, fail1.Id)
|
|
requireLinksForItem(t, m, fail2.Id, fail2Refs...)
|
|
requireItemsForLink(t, m, apiRef)
|
|
requireItemsForLink(t, m, fooRef, fail2.Id)
|
|
requireItemsForLink(t, m, barRef)
|
|
requireItemsForLink(t, m, wwwRef, fail2.Id)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc)
|
|
requireServicesTracked(t, m, fooSvc, fail2.Id)
|
|
requireServicesTracked(t, m, barSvc)
|
|
requireServicesTracked(t, m, wwwSvc, fail2.Id)
|
|
|
|
// delete the link
|
|
m.UntrackLink(newRef(fakeBarType, "www"))
|
|
|
|
requireLinksForItem(t, m, fail2.Id, newRef(fakeBarType, "foo"))
|
|
|
|
m.UntrackLink(newRef(fakeBarType, "foo"))
|
|
|
|
requireLinksForItem(t, m, fail2.Id)
|
|
|
|
// delete another item
|
|
m.UntrackItem(fail2.Id)
|
|
|
|
requireLinksForItem(t, m, fail1.Id)
|
|
requireLinksForItem(t, m, fail2.Id)
|
|
requireItemsForLink(t, m, apiRef)
|
|
requireItemsForLink(t, m, fooRef)
|
|
requireItemsForLink(t, m, barRef)
|
|
requireItemsForLink(t, m, wwwRef)
|
|
|
|
requireServicesTracked(t, m, randoSvc)
|
|
requireServicesTracked(t, m, apiSvc)
|
|
requireServicesTracked(t, m, fooSvc)
|
|
requireServicesTracked(t, m, barSvc)
|
|
requireServicesTracked(t, m, wwwSvc)
|
|
|
|
// Reset the mapper and check that its internal maps are empty.
|
|
m.Reset()
|
|
require.True(t, m.IsEmpty())
|
|
}
|
|
|
|
func TestMapper_Wildcard(t *testing.T) {
|
|
bar1Ref := newRef(fakeBarType, "uno")
|
|
bar2Ref := newRef(fakeBarType, "dos")
|
|
|
|
baz1Ref := newRef(fakeBazType, "uno")
|
|
baz2Ref := newRef(fakeBazType, "dos")
|
|
|
|
m := NewWithWildcardLinkType(fakeFooType)
|
|
|
|
foo1 := rtest.Resource(fakeFooType, "foo1").
|
|
WithTenancy(resource.DefaultNamespacedTenancy()).
|
|
Build()
|
|
foo1Refs := []resource.ReferenceOrID{
|
|
bar1Ref,
|
|
baz1Ref,
|
|
}
|
|
|
|
foo2 := rtest.Resource(fakeFooType, "foo2").
|
|
WithTenancy(resource.DefaultNamespacedTenancy()).
|
|
Build()
|
|
foo2Refs := []resource.ReferenceOrID{
|
|
bar1Ref,
|
|
bar2Ref,
|
|
baz2Ref,
|
|
}
|
|
foo2UpdatedRefs := []resource.ReferenceOrID{
|
|
bar2Ref,
|
|
baz2Ref,
|
|
}
|
|
|
|
// Nothing tracked yet so we assume nothing.
|
|
requireLinksForItem(t, m, foo1.Id)
|
|
requireLinksForItem(t, m, foo2.Id)
|
|
requireItemsForLink(t, m, bar1Ref)
|
|
requireItemsForLink(t, m, bar2Ref)
|
|
requireItemsForLink(t, m, baz1Ref)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// no-ops
|
|
m.UntrackItem(foo1.Id)
|
|
|
|
// still nothing
|
|
requireLinksForItem(t, m, foo1.Id)
|
|
requireLinksForItem(t, m, foo2.Id)
|
|
requireItemsForLink(t, m, bar1Ref)
|
|
requireItemsForLink(t, m, bar2Ref)
|
|
requireItemsForLink(t, m, baz1Ref)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// Actually insert some data.
|
|
m.TrackItem(foo1.Id, foo1Refs)
|
|
|
|
// Check links mapping
|
|
requireLinksForItem(t, m, foo1.Id, foo1Refs...)
|
|
requireItemsForLink(t, m, bar1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, bar2Ref)
|
|
requireItemsForLink(t, m, baz1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// track it again, no change
|
|
m.TrackItem(foo1.Id, foo1Refs)
|
|
|
|
requireLinksForItem(t, m, foo1.Id, foo1Refs...)
|
|
requireItemsForLink(t, m, bar1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, bar2Ref)
|
|
requireItemsForLink(t, m, baz1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// track new one that overlaps slightly
|
|
m.TrackItem(foo2.Id, foo2Refs)
|
|
|
|
// Check links mapping for the new one
|
|
requireLinksForItem(t, m, foo1.Id, foo1Refs...)
|
|
requireLinksForItem(t, m, foo2.Id, foo2Refs...)
|
|
requireItemsForLink(t, m, bar1Ref, foo1.Id, foo2.Id)
|
|
requireItemsForLink(t, m, bar2Ref, foo2.Id)
|
|
requireItemsForLink(t, m, baz1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, baz2Ref, foo2.Id)
|
|
|
|
// update the original to change it
|
|
m.TrackItem(foo2.Id, foo2UpdatedRefs)
|
|
|
|
requireLinksForItem(t, m, foo1.Id, foo1Refs...)
|
|
requireLinksForItem(t, m, foo2.Id, foo2UpdatedRefs...)
|
|
requireItemsForLink(t, m, bar1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, bar2Ref, foo2.Id)
|
|
requireItemsForLink(t, m, baz1Ref, foo1.Id)
|
|
requireItemsForLink(t, m, baz2Ref, foo2.Id)
|
|
|
|
// delete the original
|
|
m.UntrackItem(foo1.Id)
|
|
|
|
requireLinksForItem(t, m, foo1.Id)
|
|
requireLinksForItem(t, m, foo2.Id, foo2UpdatedRefs...)
|
|
requireItemsForLink(t, m, bar1Ref)
|
|
requireItemsForLink(t, m, bar2Ref, foo2.Id)
|
|
requireItemsForLink(t, m, baz1Ref)
|
|
requireItemsForLink(t, m, baz2Ref, foo2.Id)
|
|
|
|
// delete the link
|
|
m.UntrackLink(baz2Ref)
|
|
|
|
foo2DoubleUpdatedRefs := []resource.ReferenceOrID{
|
|
bar2Ref,
|
|
}
|
|
|
|
requireLinksForItem(t, m, foo1.Id)
|
|
requireLinksForItem(t, m, foo2.Id, foo2DoubleUpdatedRefs...)
|
|
requireItemsForLink(t, m, bar1Ref)
|
|
requireItemsForLink(t, m, bar2Ref, foo2.Id)
|
|
requireItemsForLink(t, m, baz1Ref)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// delete another item
|
|
m.UntrackItem(foo2.Id)
|
|
|
|
requireLinksForItem(t, m, foo1.Id)
|
|
requireLinksForItem(t, m, foo2.Id)
|
|
requireItemsForLink(t, m, bar1Ref)
|
|
requireItemsForLink(t, m, bar2Ref)
|
|
requireItemsForLink(t, m, baz1Ref)
|
|
requireItemsForLink(t, m, baz2Ref)
|
|
|
|
// Reset the mapper and check that its internal maps are empty.
|
|
m.Reset()
|
|
require.True(t, m.IsEmpty())
|
|
}
|
|
|
|
func TestPanics(t *testing.T) {
|
|
t.Run("new mapper without types", func(t *testing.T) {
|
|
require.PanicsWithValue(t, "itemType is required", func() {
|
|
New(nil, nil)
|
|
})
|
|
|
|
require.PanicsWithValue(t, "itemType is required", func() {
|
|
New(nil, fakeBarType)
|
|
})
|
|
|
|
require.PanicsWithValue(t, "linkType is required", func() {
|
|
New(fakeFooType, nil)
|
|
})
|
|
})
|
|
|
|
t.Run("UntrackItem: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected item type \"catalog.v1.Foo\" got \"catalog.v1.Bar\"", func() {
|
|
// Calling UntrackItem with link type instead of item type
|
|
m.UntrackItem(rtest.Resource(fakeBarType, "test").ID())
|
|
})
|
|
})
|
|
|
|
t.Run("TrackItem: mismatched item type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected item type \"catalog.v1.Foo\" got \"catalog.v1.Bar\"", func() {
|
|
// Calling UntrackItem with link type instead of item type
|
|
m.TrackItem(rtest.Resource(fakeBarType, "test").ID(), nil)
|
|
})
|
|
})
|
|
|
|
t.Run("TrackItem: mismatched link type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected link type \"catalog.v1.Bar\" got \"catalog.v1.Foo\"", func() {
|
|
// Calling UntrackItem with link type instead of item type
|
|
links := []resource.ReferenceOrID{
|
|
rtest.Resource(fakeFooType, "link").ID(),
|
|
}
|
|
m.TrackItem(rtest.Resource(fakeFooType, "test").ID(), links)
|
|
})
|
|
})
|
|
|
|
t.Run("UntrackLink: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected link type \"catalog.v1.Bar\" got \"catalog.v1.Foo\"", func() {
|
|
m.UntrackLink(rtest.Resource(fakeFooType, "test").ID())
|
|
})
|
|
})
|
|
|
|
t.Run("LinkRefsForItem: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected item type \"catalog.v1.Foo\" got \"catalog.v1.Bar\"", func() {
|
|
m.LinkRefsForItem(rtest.Resource(fakeBarType, "test").ID())
|
|
})
|
|
})
|
|
|
|
t.Run("LinkRefsForItem: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected item type \"catalog.v1.Foo\" got \"catalog.v1.Bar\"", func() {
|
|
m.LinkIDsForItem(rtest.Resource(fakeBarType, "test").ID())
|
|
})
|
|
})
|
|
|
|
t.Run("ItemRefsForLink: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected link type \"catalog.v1.Bar\" got \"catalog.v1.Foo\"", func() {
|
|
m.ItemRefsForLink(rtest.Resource(fakeFooType, "test").ID())
|
|
})
|
|
})
|
|
|
|
t.Run("ItemIDsForLink: mismatched type", func(t *testing.T) {
|
|
m := New(fakeFooType, fakeBarType)
|
|
require.PanicsWithValue(t, "expected link type \"catalog.v1.Bar\" got \"catalog.v1.Foo\"", func() {
|
|
m.ItemIDsForLink(rtest.Resource(fakeFooType, "test").ID())
|
|
})
|
|
})
|
|
}
|
|
|
|
func requireServicesTracked(t *testing.T, mapper *Mapper, link *pbresource.Resource, items ...*pbresource.ID) {
|
|
t.Helper()
|
|
|
|
reqs, err := mapper.MapLink(
|
|
context.Background(),
|
|
controller.Runtime{},
|
|
link,
|
|
)
|
|
require.NoError(t, err)
|
|
|
|
require.Len(t, reqs, len(items))
|
|
|
|
// Also check items IDs and Refs for link.
|
|
ids := mapper.ItemIDsForLink(link.Id)
|
|
require.Len(t, ids, len(items))
|
|
|
|
refs := mapper.ItemRefsForLink(link.Id)
|
|
require.Len(t, refs, len(items))
|
|
|
|
for _, item := range items {
|
|
prototest.AssertContainsElement(t, reqs, controller.Request{ID: item})
|
|
prototest.AssertContainsElement(t, ids, item)
|
|
prototest.AssertContainsElement(t, refs, resource.Reference(item, ""))
|
|
}
|
|
}
|
|
|
|
func requireItemsForLink(t *testing.T, mapper *Mapper, link *pbresource.Reference, items ...*pbresource.ID) {
|
|
t.Helper()
|
|
|
|
got := mapper.ItemIDsForLink(resource.IDFromReference(link))
|
|
|
|
prototest.AssertElementsMatch(t, items, got)
|
|
}
|
|
|
|
func requireLinksForItem(t *testing.T, mapper *Mapper, item *pbresource.ID, links ...resource.ReferenceOrID) {
|
|
t.Helper()
|
|
|
|
var expLinkRefs []*pbresource.Reference
|
|
var expLinkIDs []*pbresource.ID
|
|
|
|
for _, l := range links {
|
|
expLinkRefs = append(expLinkRefs, &pbresource.Reference{
|
|
Name: l.GetName(),
|
|
Tenancy: l.GetTenancy(),
|
|
Type: l.GetType(),
|
|
})
|
|
expLinkIDs = append(expLinkIDs, &pbresource.ID{
|
|
Name: l.GetName(),
|
|
Tenancy: l.GetTenancy(),
|
|
Type: l.GetType(),
|
|
})
|
|
}
|
|
|
|
refs := mapper.LinkRefsForItem(item)
|
|
require.Len(t, refs, len(links))
|
|
prototest.AssertElementsMatch(t, expLinkRefs, refs)
|
|
|
|
ids := mapper.LinkIDsForItem(item)
|
|
require.Len(t, refs, len(links))
|
|
prototest.AssertElementsMatch(t, expLinkIDs, ids)
|
|
}
|
|
|
|
func newRef(typ *pbresource.Type, name string) *pbresource.Reference {
|
|
return rtest.Resource(typ, name).WithTenancy(resource.DefaultNamespacedTenancy()).Reference("")
|
|
}
|