Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
137738: sql: fix including all related FK tables into the stmt bundle r=yuzefovich a=yuzefovich

Earlier this year in 084a741 we made a change to include all tables referencing the table from the stmt bundle (i.e. those that we have an inbound FK relationship with). However, we forgot to include tables that the referencing table is dependent on, so this commit fixes that oversight. Namely, we now include all tables that we reference or that reference us via the FK relationship, and this rule is applied recursively to every table in consideration. (Previously, we didn't include inbound FKs when handling an outbound FK nor outbound FKs when handling an inbound FK.)

Note that I believe we could have avoided including referencing tables (i.e. with an inbound FK relationship) that only have NO ACTION or RESTRICT actions in the ON DELETE and ON UPDATE, but I think those cases aren't very common, and it's unlikely to hurt including all "related" tables. The case of FK cycles is still not handled correctly on the stmt bundle recreation, but I think we have yet to run into one in a support ticket.

Informs: https://github.com/cockroachlabs/support/issues/3146.
Epic: None

Release note: None

137739: sql: remove unused dropOwnedByNode r=mgartner a=mgartner

`DROP OWNED BY` is supported only in the declarative schema changer and
`dropOwnedByNode` is not used, so it has been removed.

Epic: None

Release note: None


137742: roachtest: use correct warehouse number in import r=yuzefovich a=yuzefovich

In 0a94874 we introduced a bug which hard-coded that we'd always use 1 warehouse in tpcc import roachtests. This is now fixed.

Epic: None

Release note: None

137744: sql: allow CLOSE CURSOR in read-only txns r=yuzefovich a=yuzefovich

This is allowed in PG, so we now will allow this too. The fix is similar to what we did for DECLARE and FETCH in 50a7999.

Fixes: #137606.

Release note (bug fix): CLOSE CURSOR statements are now allowed in read-only transactions, similat to Postgres. The bug has been present since at least 23.1 version.

Co-authored-by: Yahor Yuzefovich <[email protected]>
Co-authored-by: Marcus Gartner <[email protected]>
  • Loading branch information
3 people committed Dec 19, 2024
5 parents 6087ddf + 982abc2 + abbbbc2 + c06ba73 + f09d974 commit a833dc2
Show file tree
Hide file tree
Showing 7 changed files with 47 additions and 59 deletions.
2 changes: 1 addition & 1 deletion pkg/cmd/roachtest/tests/import.go
Original file line number Diff line number Diff line change
Expand Up @@ -150,7 +150,7 @@ func registerImportTPCC(r registry.Registry) {
} else {
defer hc.Done()
}
cmd := fmt.Sprintf(workloadStr, 1)
cmd := fmt.Sprintf(workloadStr, warehouses)
// Tick once before starting the import, and once after to capture the
// total elapsed time. This is used by roachperf to compute and display
// the average MB/sec per node.
Expand Down
15 changes: 0 additions & 15 deletions pkg/sql/drop_owned_by.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,17 +9,10 @@ import (
"context"

"github.com/cockroachdb/cockroach/pkg/server/telemetry"
"github.com/cockroachdb/cockroach/pkg/sql/sem/tree"
"github.com/cockroachdb/cockroach/pkg/sql/sqltelemetry"
"github.com/cockroachdb/cockroach/pkg/util/errorutil/unimplemented"
)

// dropOwnedByNode represents a DROP OWNED BY <role(s)> statement.
type dropOwnedByNode struct {
// TODO(angelaw): Uncomment when implementing - commenting out due to linting error.
//n *tree.DropOwnedBy
}

func (p *planner) DropOwnedBy(ctx context.Context) (planNode, error) {
if err := checkSchemaChangeEnabled(
ctx,
Expand All @@ -32,11 +25,3 @@ func (p *planner) DropOwnedBy(ctx context.Context) (planNode, error) {
// TODO(angelaw): Implementation.
return nil, unimplemented.NewWithIssue(55381, "drop owned by is not yet implemented")
}

func (n *dropOwnedByNode) startExec(params runParams) error {
// TODO(angelaw): Implementation.
return nil
}
func (n *dropOwnedByNode) Next(runParams) (bool, error) { return false, nil }
func (n *dropOwnedByNode) Values() tree.Datums { return tree.Datums{} }
func (n *dropOwnedByNode) Close(context.Context) {}
17 changes: 12 additions & 5 deletions pkg/sql/explain_bundle_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -276,11 +276,18 @@ CREATE TABLE users(id UUID DEFAULT gen_random_uuid() PRIMARY KEY, promo_id INT R
})

t.Run("foreign keys", func(t *testing.T) {
// All tables should be included in the stmt bundle, regardless of which
// one we query because all of them are considered "related" (even
// though we don't specify ON DELETE and ON UPDATE actions).
tableNames := []string{"parent", "child1", "child2", "grandchild1", "grandchild2"}
r.Exec(t, "CREATE TABLE parent (pk INT PRIMARY KEY, v INT);")
r.Exec(t, "CREATE TABLE child (pk INT PRIMARY KEY, fk INT REFERENCES parent(pk));")
r.Exec(t, "CREATE TABLE child1 (pk INT PRIMARY KEY, fk INT REFERENCES parent(pk));")
r.Exec(t, "CREATE TABLE child2 (pk INT PRIMARY KEY, fk INT REFERENCES parent(pk));")
r.Exec(t, "CREATE TABLE grandchild1 (pk INT PRIMARY KEY, fk INT REFERENCES child1(pk));")
r.Exec(t, "CREATE TABLE grandchild2 (pk INT PRIMARY KEY, fk INT REFERENCES child2(pk));")
contentCheck := func(name, contents string) error {
if name == "schema.sql" {
for _, tableName := range []string{"parent", "child"} {
for _, tableName := range tableNames {
if regexp.MustCompile("CREATE TABLE defaultdb.public."+tableName).FindString(contents) == "" {
return errors.Newf(
"could not find 'CREATE TABLE defaultdb.public.%s' in schema.sql:\n%s", tableName, contents)
Expand All @@ -289,12 +296,12 @@ CREATE TABLE users(id UUID DEFAULT gen_random_uuid() PRIMARY KEY, promo_id INT R
}
return nil
}
for _, tableName := range []string{"parent", "child"} {
for _, tableName := range tableNames {
rows := r.QueryStr(t, "EXPLAIN ANALYZE (DEBUG) SELECT * FROM "+tableName)
checkBundle(
t, fmt.Sprint(rows), "child", contentCheck, false, /* expectErrors */
base, plans, "stats-defaultdb.public.parent.sql", "stats-defaultdb.public.child.sql",
"distsql.html vec.txt vec-v.txt",
base, plans, "stats-defaultdb.public.parent.sql", "stats-defaultdb.public.child1.sql", "stats-defaultdb.public.child2.sql",
"stats-defaultdb.public.grandchild1.sql", "stats-defaultdb.public.grandchild2.sql", "distsql.html vec.txt vec-v.txt",
)
}
})
Expand Down
5 changes: 4 additions & 1 deletion pkg/sql/logictest/testdata/logic_test/txn
Original file line number Diff line number Diff line change
Expand Up @@ -1520,13 +1520,16 @@ SET SESSION AUTHORIZATION DEFAULT
statement ok
BEGIN

# DECLARE and FETCH CURSOR should work in a read-only txn.
# DECLARE, FETCH, and CLOSE CURSOR should work in a read-only txn.
statement ok
DECLARE foo CURSOR FOR SELECT 1

statement ok
FETCH 1 foo

statement ok
CLOSE foo

statement ok
COMMIT

Expand Down
64 changes: 29 additions & 35 deletions pkg/sql/opt/metadata.go
Original file line number Diff line number Diff line change
Expand Up @@ -1007,49 +1007,43 @@ func (md *Metadata) getAllReferenceTables(
var tableSet intsets.Fast
var tableList []cat.DataSource
var addForeignKeyReferencedTables func(tab cat.Table)
var addForeignKeyReferencingTables func(tab cat.Table)
// handleRelatedTables is a helper function that processes the given table
// if it hasn't been handled yet by adding all referenced and referencing
// table of the given one, including via transient (recursive) FK
// relationships.
handleRelatedTables := func(tabID cat.StableID) {
if !tableSet.Contains(int(tabID)) {
tableSet.Add(int(tabID))
ds, _, err := catalog.ResolveDataSourceByID(ctx, cat.Flags{}, tabID)
if err != nil {
// This is a best-effort attempt to get all the tables, so don't
// error.
return
}
refTab, ok := ds.(cat.Table)
if !ok {
// This is a best-effort attempt to get all the tables, so don't
// error.
return
}
// We want to include all tables that we reference before adding
// ourselves, followed by all tables that reference us.
addForeignKeyReferencedTables(refTab)
tableList = append(tableList, ds)
addForeignKeyReferencingTables(refTab)
}
}
addForeignKeyReferencedTables = func(tab cat.Table) {
for i := 0; i < tab.OutboundForeignKeyCount(); i++ {
tabID := tab.OutboundForeignKey(i).ReferencedTableID()
if !tableSet.Contains(int(tabID)) {
tableSet.Add(int(tabID))
ds, _, err := catalog.ResolveDataSourceByID(ctx, cat.Flags{}, tabID)
if err != nil {
// This is a best-effort attempt to get all the tables, so don't error.
continue
}
refTab, ok := ds.(cat.Table)
if !ok {
// This is a best-effort attempt to get all the tables, so don't error.
continue
}
// We want to include all tables that we reference before adding
// ourselves.
addForeignKeyReferencedTables(refTab)
tableList = append(tableList, ds)
}
handleRelatedTables(tabID)
}
}
var addForeignKeyReferencingTables func(tab cat.Table)
addForeignKeyReferencingTables = func(tab cat.Table) {
for i := 0; i < tab.InboundForeignKeyCount(); i++ {
tabID := tab.InboundForeignKey(i).OriginTableID()
if !tableSet.Contains(int(tabID)) {
tableSet.Add(int(tabID))
ds, _, err := catalog.ResolveDataSourceByID(ctx, cat.Flags{}, tabID)
if err != nil {
// This is a best-effort attempt to get all the tables, so don't error.
continue
}
refTab, ok := ds.(cat.Table)
if !ok {
// This is a best-effort attempt to get all the tables, so don't error.
continue
}
// We want to include ourselves before all tables that reference
// us.
tableList = append(tableList, ds)
addForeignKeyReferencingTables(refTab)
}
handleRelatedTables(tabID)
}
}
for i := range md.tables {
Expand Down
2 changes: 1 addition & 1 deletion pkg/sql/sem/tree/stmt.go
Original file line number Diff line number Diff line change
Expand Up @@ -768,7 +768,7 @@ func (*CannedOptPlan) StatementTag() string { return "PREPARE AS OPT PLAN" }
func (*CloseCursor) StatementReturnType() StatementReturnType { return Ack }

// StatementType implements the Statement interface.
func (*CloseCursor) StatementType() StatementType { return TypeDCL }
func (*CloseCursor) StatementType() StatementType { return TypeDML }

// StatementTag returns a short string identifying the type of statement.
func (*CloseCursor) StatementTag() string { return "CLOSE" }
Expand Down
1 change: 0 additions & 1 deletion pkg/sql/walk.go
Original file line number Diff line number Diff line change
Expand Up @@ -432,7 +432,6 @@ var planNodeNames = map[reflect.Type]string{
reflect.TypeOf(&ordinalityNode{}): "ordinality",
reflect.TypeOf(&projectSetNode{}): "project set",
reflect.TypeOf(&reassignOwnedByNode{}): "reassign owned by",
reflect.TypeOf(&dropOwnedByNode{}): "drop owned by",
reflect.TypeOf(&recursiveCTENode{}): "recursive cte",
reflect.TypeOf(&refreshMaterializedViewNode{}): "refresh materialized view",
reflect.TypeOf(&relocateNode{}): "relocate",
Expand Down

0 comments on commit a833dc2

Please sign in to comment.