Skip to content

Commit

Permalink
sql/opt: fix negative annotation caching for RegionConfig
Browse files Browse the repository at this point in the history
Fixes #88511

Release note (performance improvement): Fixed minor bug where negative caching
of region configuration information could result in extra work when use a
database that is not configured for multi-region.
  • Loading branch information
ajwerner committed Sep 22, 2022
1 parent 9c0778f commit f42e0ee
Show file tree
Hide file tree
Showing 4 changed files with 65 additions and 6 deletions.
2 changes: 2 additions & 0 deletions pkg/sql/opt/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,7 @@ go_test(
embed = [":opt"],
deps = [
"//pkg/settings/cluster",
"//pkg/sql/catalog/descpb",
"//pkg/sql/opt/cat",
"//pkg/sql/opt/memo",
"//pkg/sql/opt/norm",
Expand All @@ -70,6 +71,7 @@ go_test(
"//pkg/sql/sem/tree",
"//pkg/sql/types",
"//pkg/util",
"@com_github_stretchr_testify//require",
],
)

Expand Down
46 changes: 46 additions & 0 deletions pkg/sql/opt/metadata_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import (
"testing"

"github.com/cockroachdb/cockroach/pkg/settings/cluster"
"github.com/cockroachdb/cockroach/pkg/sql/catalog/descpb"
"github.com/cockroachdb/cockroach/pkg/sql/opt"
"github.com/cockroachdb/cockroach/pkg/sql/opt/cat"
"github.com/cockroachdb/cockroach/pkg/sql/opt/memo"
Expand All @@ -26,6 +27,7 @@ import (
"github.com/cockroachdb/cockroach/pkg/sql/sem/eval"
"github.com/cockroachdb/cockroach/pkg/sql/sem/tree"
"github.com/cockroachdb/cockroach/pkg/sql/types"
"github.com/stretchr/testify/require"
)

func TestMetadata(t *testing.T) {
Expand Down Expand Up @@ -409,3 +411,47 @@ func TestDuplicateTable(t *testing.T) {
t.Errorf("expected partial index predicate to reference new column ID %d, got %d", dupB, col)
}
}

// TestTableMeta_GetRegionsInDatabases exercises the multiregion.RegionConfig
// annotation.
func TestTableMeta_GetRegionsInDatabase(t *testing.T) {
cat := testcat.New()
_, err := cat.ExecuteDDL("CREATE TABLE a (b BOOL, b2 BOOL, INDEX (b2) WHERE b)")
if err != nil {
t.Fatal(err)
}

var md opt.Metadata
tn := tree.NewUnqualifiedTableName("a")
tab := cat.Table(tn)
tab.DatabaseID = 1 // must be non-zero to trigger the region lookup
a := md.AddTable(tab, tn)
tabMeta := md.TableMeta(a)

p := &fakeGetMultiregionConfigPlanner{}
// Call the function once, make sure our planner method gets invoked.
{
_, exists := tabMeta.GetRegionsInDatabase(p)
require.False(t, exists)
require.Equal(t, 1, p.getMultiregionConfigCalled)
}
// Call the function again, make sure that our planner method doesn't
// get invoked again.
{
_, exists := tabMeta.GetRegionsInDatabase(p)
require.False(t, exists)
require.Equal(t, 1, p.getMultiregionConfigCalled)
}
}

type fakeGetMultiregionConfigPlanner struct {
eval.Planner
getMultiregionConfigCalled int
}

func (f *fakeGetMultiregionConfigPlanner) GetMultiregionConfig(
databaseID descpb.ID,
) (interface{}, bool) {
f.getMultiregionConfigCalled++
return nil, false
}
20 changes: 15 additions & 5 deletions pkg/sql/opt/table_meta.go
Original file line number Diff line number Diff line change
Expand Up @@ -449,7 +449,7 @@ func (tm *TableMeta) VirtualComputedColumns() ColSet {
// multiregion. The result is cached in TableMeta.
func (tm *TableMeta) GetRegionsInDatabase(
planner eval.Planner,
) (regionNames catpb.RegionNames, ok bool) {
) (regionNames catpb.RegionNames, hasRegionNames bool) {
multiregionConfig, ok := tm.TableAnnotation(regionConfigAnnID).(*multiregion.RegionConfig)
if ok {
if multiregionConfig == nil {
Expand All @@ -458,14 +458,22 @@ func (tm *TableMeta) GetRegionsInDatabase(
return multiregionConfig.Regions(), true
}
dbID := tm.Table.GetDatabaseID()
defer func() {
if !hasRegionNames {
tm.SetTableAnnotation(
regionConfigAnnID,
// Use a nil pointer to a RegionConfig, which is distinct from the
// untyped nil and will be detected in the type assertion above.
(*multiregion.RegionConfig)(nil),
)
}
}()

if dbID == 0 {
tm.SetTableAnnotation(regionConfigAnnID, nil)
return nil /* regionNames */, false
}

regionConfig, ok := planner.GetMultiregionConfig(dbID)
if !ok {
tm.SetTableAnnotation(regionConfigAnnID, nil)
return nil /* regionNames */, false
}
multiregionConfig, _ = regionConfig.(*multiregion.RegionConfig)
Expand Down Expand Up @@ -494,7 +502,9 @@ func (tm *TableMeta) GetDatabaseSurvivalGoal(
dbID := tm.Table.GetDatabaseID()
regionConfig, ok := planner.GetMultiregionConfig(dbID)
if !ok {
tm.SetTableAnnotation(regionConfigAnnID, nil)
// Use a nil pointer to a RegionConfig, which is distinct from the
// untyped nil and will be detected in the type assertion above.
tm.SetTableAnnotation(regionConfigAnnID, (*multiregion.RegionConfig)(nil))
return descpb.SurvivalGoal_ZONE_FAILURE /* survivalGoal */, false
}
multiregionConfig, _ = regionConfig.(*multiregion.RegionConfig)
Expand Down
3 changes: 2 additions & 1 deletion pkg/sql/opt/testutils/testcat/test_catalog.go
Original file line number Diff line number Diff line change
Expand Up @@ -667,6 +667,7 @@ func (tv *View) CollectTypes(ord int) (descpb.IDs, error) {
// Table implements the cat.Table interface for testing purposes.
type Table struct {
TabID cat.StableID
DatabaseID descpb.ID
TabVersion int
TabName tree.TableName
Columns []cat.Column
Expand Down Expand Up @@ -869,7 +870,7 @@ func (tt *Table) HomeRegionColName() (colName string, ok bool) {

// GetDatabaseID is part of the cat.Table interface.
func (tt *Table) GetDatabaseID() descpb.ID {
return 0
return tt.DatabaseID
}

// FindOrdinal returns the ordinal of the column with the given name.
Expand Down

0 comments on commit f42e0ee

Please sign in to comment.