From 4ba58dc1d93bfe8b41ee4b8b0beefe8e0e537a73 Mon Sep 17 00:00:00 2001 From: Andrew Werner Date: Thu, 21 Jan 2021 22:38:18 -0500 Subject: [PATCH] sql: prevent DROP SCHEMA CASCADE from droping types with references Before this patch, a DROP SCHEMA CASCADE could cause database corruption by dropping types which were referenced by other tables. This would lead to bad errors like: ``` ERROR: object already exists: desc 687: type ID 685 in descriptor not found: descriptor not found SQLSTATE: 42710 ``` And doctor errors like: ``` Table 687: ParentID 50, ParentSchemaID 29, Name 't': type ID 685 in descriptor not found: descriptor not found ``` Fixes #59021. Release note (bug fix): Fixed a bug where `DROP SCHEMA ... CASCADE` could result in types which are referenced being dropped. --- pkg/sql/drop_cascade.go | 40 +++++++++++++++++++ pkg/sql/drop_type.go | 14 ++----- .../logictest/testdata/logic_test/drop_type | 21 ++++++++++ pkg/sql/resolver.go | 26 ++++++++++++ 4 files changed, 90 insertions(+), 11 deletions(-) diff --git a/pkg/sql/drop_cascade.go b/pkg/sql/drop_cascade.go index 52529b44f9ab..c04ada462af5 100644 --- a/pkg/sql/drop_cascade.go +++ b/pkg/sql/drop_cascade.go @@ -23,6 +23,7 @@ import ( "github.com/cockroachdb/cockroach/pkg/sql/pgwire/pgcode" "github.com/cockroachdb/cockroach/pkg/sql/pgwire/pgerror" "github.com/cockroachdb/cockroach/pkg/sql/sem/tree" + "github.com/cockroachdb/cockroach/pkg/util/errorutil/unimplemented" "github.com/cockroachdb/errors" ) @@ -32,6 +33,7 @@ type dropCascadeState struct { objectNamesToDelete []tree.ObjectName td []toDelete + toDeleteByID map[descpb.ID]*toDelete allTableObjectsToDelete []*tabledesc.Mutable typesToDelete []*typedesc.Mutable @@ -176,6 +178,10 @@ func (d *dropCascadeState) resolveCollectedObjects( } d.allTableObjectsToDelete = allObjectsToDelete d.td = filterImplicitlyDeletedObjects(d.td, implicitDeleteMap) + d.toDeleteByID = make(map[descpb.ID]*toDelete) + for i := range d.td { + d.toDeleteByID[d.td[i].desc.GetID()] = &d.td[i] + } return nil } @@ -203,6 +209,9 @@ func (d *dropCascadeState) dropAllCollectedObjects(ctx context.Context, p *plann // Now delete all of the types. for _, typ := range d.typesToDelete { + if err := d.canDropType(ctx, p, typ); err != nil { + return err + } // Drop the types. Note that we set queueJob to be false because the types // will be dropped in bulk as part of the DROP DATABASE job. if err := p.dropTypeImpl(ctx, typ, "", false /* queueJob */); err != nil { @@ -213,6 +222,37 @@ func (d *dropCascadeState) dropAllCollectedObjects(ctx context.Context, p *plann return nil } +func (d *dropCascadeState) canDropType( + ctx context.Context, p *planner, typ *typedesc.Mutable, +) error { + var referencedButNotDropping []descpb.ID + for _, id := range typ.ReferencingDescriptorIDs { + if _, exists := d.toDeleteByID[id]; exists { + continue + } + referencedButNotDropping = append(referencedButNotDropping, id) + } + if len(referencedButNotDropping) == 0 { + return nil + } + dependentNames, err := p.getFullyQualifiedTableNamesFromIDs(ctx, referencedButNotDropping) + if err != nil { + return errors.Wrapf(err, "type %q has dependent objects", typ.Name) + } + fqName, err := getTypeNameFromTypeDescriptor( + oneAtATimeSchemaResolver{ctx, p}, + typ, + ) + if err != nil { + return errors.Wrapf(err, "type %q has dependent objects", typ.Name) + } + return unimplemented.NewWithIssueDetailf(51480, "DROP TYPE CASCADE is not yet supported", + "cannot drop type %q because other objects (%v) still depend on it", + fqName.FQString(), + dependentNames, + ) +} + func (d *dropCascadeState) getDroppedTableDetails() []jobspb.DroppedTableDetails { res := make([]jobspb.DroppedTableDetails, len(d.allTableObjectsToDelete)) for i := range d.allTableObjectsToDelete { diff --git a/pkg/sql/drop_type.go b/pkg/sql/drop_type.go index d04a9d6d970c..377773826c76 100644 --- a/pkg/sql/drop_type.go +++ b/pkg/sql/drop_type.go @@ -133,17 +133,9 @@ func (p *planner) canDropTypeDesc( return err } if len(desc.ReferencingDescriptorIDs) > 0 && behavior != tree.DropCascade { - var dependentNames []string - for _, id := range desc.ReferencingDescriptorIDs { - desc, err := p.Descriptors().GetMutableTableVersionByID(ctx, id, p.txn) - if err != nil { - return errors.Wrapf(err, "type has dependent objects") - } - fqName, err := p.getQualifiedTableName(ctx, desc) - if err != nil { - return errors.Wrapf(err, "type %q has dependent objects", desc.Name) - } - dependentNames = append(dependentNames, fqName.FQString()) + dependentNames, err := p.getFullyQualifiedTableNamesFromIDs(ctx, desc.ReferencingDescriptorIDs) + if err != nil { + return errors.Wrapf(err, "type %q has dependent objects", desc.Name) } return pgerror.Newf( pgcode.DependentObjectsStillExist, diff --git a/pkg/sql/logictest/testdata/logic_test/drop_type b/pkg/sql/logictest/testdata/logic_test/drop_type index c8e97bf0f2ed..4d1fad66c5ae 100644 --- a/pkg/sql/logictest/testdata/logic_test/drop_type +++ b/pkg/sql/logictest/testdata/logic_test/drop_type @@ -351,3 +351,24 @@ DROP TYPE d."b+c" statement ok DROP DATABASE d + +# Test that dropping a schema which contains a type which refers to things +# outside of that schema works. + +subtest drop_schema_cascade + +statement ok +CREATE SCHEMA schema_to_drop; +CREATE TYPE schema_to_drop.typ AS ENUM ('a'); +CREATE TABLE t (k schema_to_drop.typ PRIMARY KEY); +CREATE TABLE schema_to_drop.t (k schema_to_drop.typ PRIMARY KEY); + +statement error pgcode 0A000 unimplemented: cannot drop type "test.schema_to_drop._typ" because other objects \(\[test\.public\.t\]\) still depend on it +DROP SCHEMA schema_to_drop CASCADE; + +statement ok +DROP TABLE t; + +statement ok +DROP SCHEMA schema_to_drop CASCADE; + diff --git a/pkg/sql/resolver.go b/pkg/sql/resolver.go index 8306ab18132e..666ef15be0d8 100644 --- a/pkg/sql/resolver.go +++ b/pkg/sql/resolver.go @@ -398,6 +398,32 @@ func getDescriptorsFromTargetListForPrivilegeChange( return descs, nil } +// getFullyQualifiedTableNamesFromIDs resolves a list of table IDs to their +// fully qualified names. +func (p *planner) getFullyQualifiedTableNamesFromIDs( + ctx context.Context, ids []descpb.ID, +) (fullyQualifiedNames []*tree.TableName, _ error) { + for _, id := range ids { + desc, err := p.Descriptors().GetImmutableTableByID(ctx, p.txn, id, tree.ObjectLookupFlags{ + CommonLookupFlags: tree.CommonLookupFlags{ + Required: true, + AvoidCached: true, + IncludeDropped: true, + IncludeOffline: true, + }, + }) + if err != nil { + return nil, err + } + fqName, err := p.getQualifiedTableName(ctx, desc) + if err != nil { + return nil, err + } + fullyQualifiedNames = append(fullyQualifiedNames, fqName) + } + return fullyQualifiedNames, nil +} + // getQualifiedTableName returns the database-qualified name of the table // or view represented by the provided descriptor. It is a sort of // reverse of the Resolve() functions.