Skip to content

Commit

Permalink
schemadiff: EnumReorderStrategy, checking if enum or set values c…
Browse files Browse the repository at this point in the history
…hange ordinal (#15106)

Signed-off-by: Shlomi Noach <[email protected]>
  • Loading branch information
shlomi-noach authored Feb 5, 2024
1 parent b6d5655 commit e9c513e
Show file tree
Hide file tree
Showing 27 changed files with 376 additions and 100 deletions.
3 changes: 3 additions & 0 deletions go/sqlescape/ids_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,9 @@ func TestEscapeID(t *testing.T) {
}, {
in: "a`a",
out: "`a``a`",
}, {
in: "a a",
out: "`a a`",
}, {
in: "`fo`o`",
out: "```fo``o```",
Expand Down
4 changes: 2 additions & 2 deletions go/test/endtoend/onlineddl/revert/onlineddl_revert_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -357,8 +357,8 @@ func testRevertible(t *testing.T) {
},
{
name: "expanded: enum",
fromSchema: `id int primary key, e1 enum('a', 'b'), e2 enum('a', 'b'), e3 enum('a', 'b'), e4 enum('a', 'b'), e5 enum('a', 'b'), e6 enum('a', 'b'), e7 enum('a', 'b'), e8 enum('a', 'b')`,
toSchema: `id int primary key, e1 enum('a', 'b'), e2 enum('a'), e3 enum('a', 'b', 'c'), e4 enum('a', 'x'), e5 enum('a', 'x', 'b'), e6 enum('b'), e7 varchar(1), e8 tinyint`,
fromSchema: `id int primary key, e1 enum('a', 'b'), e2 enum('a', 'b'), e3 enum('a', 'b'), e4 enum('a', 'b'), e5 enum('a', 'b'), e6 enum('a', 'b'), e7 enum('a', 'b'), e8 enum('a', 'b')`,
toSchema: `id int primary key, e1 enum('a', 'b'), e2 enum('a'), e3 enum('a', 'b', 'c'), e4 enum('a', 'x'), e5 enum('a', 'x', 'b'), e6 enum('b'), e7 varchar(1), e8 tinyint`,
expandedColumnNames: `e3,e4,e5,e6,e7,e8`,
},
{
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green', 'cyan') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');

drop event if exists onlineddl_test;
delimiter ;;
create event onlineddl_test
on schedule every 1 second
starts current_timestamp
ends current_timestamp + interval 60 second
on completion not preserve
enable
do
begin
insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
end ;;
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');

drop event if exists onlineddl_test;
delimiter ;;
create event onlineddl_test
on schedule every 1 second
starts current_timestamp
ends current_timestamp + interval 60 second
on completion not preserve
enable
do
begin
insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
end ;;
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green', 'cyan') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

drop event if exists onlineddl_test;
delimiter ;;
create event onlineddl_test
on schedule every 1 second
starts current_timestamp
ends current_timestamp + interval 60 second
on completion not preserve
enable
do
begin
insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
insert into onlineddl_test values (null, 17, 'blue');
end ;;
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Data truncated for column
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green', 'cyan') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
insert into onlineddl_test values (null, 17, 'blue');
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Data truncated for column
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

drop event if exists onlineddl_test;
delimiter ;;
create event onlineddl_test
on schedule every 1 second
starts current_timestamp
ends current_timestamp + interval 60 second
on completion not preserve
enable
do
begin
insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
insert into onlineddl_test values (null, 17, 'blue');
end ;;
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Data truncated for column
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
modify e enum('red', 'green') not null
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
drop table if exists onlineddl_test;
create table onlineddl_test (
id int auto_increment,
i int not null,
e enum('red', 'green', 'blue') not null,
primary key(id)
) auto_increment=1;

insert into onlineddl_test values (null, 11, 'red');
insert into onlineddl_test values (null, 13, 'green');
insert into onlineddl_test values (null, 17, 'blue');
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
Data truncated for column
31 changes: 24 additions & 7 deletions go/vt/schemadiff/column.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,6 @@ import (

"vitess.io/vitess/go/mysql/collations"
"vitess.io/vitess/go/vt/sqlparser"
"vitess.io/vitess/go/vt/vterrors"

vtrpcpb "vitess.io/vitess/go/vt/proto/vtrpc"
)

// columnDetails decorates a column with more details, used by diffing logic
Expand Down Expand Up @@ -99,9 +96,11 @@ func NewColumnDefinitionEntity(c *sqlparser.ColumnDefinition) *ColumnDefinitionE
// We need to denormalize the column's charset/collate properties, so that the comparison can be done.
func (c *ColumnDefinitionEntity) ColumnDiff(
env *Environment,
tableName string,
other *ColumnDefinitionEntity,
t1cc *charsetCollate,
t2cc *charsetCollate,
hints *DiffHints,
) (*ModifyColumnDiff, error) {
if c.IsTextual() || other.IsTextual() {
// We will now denormalize the columns charset & collate as needed (if empty, populate from table.)
Expand All @@ -111,7 +110,7 @@ func (c *ColumnDefinitionEntity) ColumnDiff(
collationID := env.CollationEnv().LookupByName(c.columnDefinition.Type.Options.Collate)
charset := env.CollationEnv().LookupCharsetName(collationID)
if charset == "" {
return nil, vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "cannot match charset to collation %v", c.columnDefinition.Type.Options.Collate)
return nil, &UnknownColumnCollationCharsetError{Column: c.columnDefinition.Name.String(), Collation: c.columnDefinition.Type.Options.Collate}
}
defer func() {
c.columnDefinition.Type.Charset.Name = ""
Expand All @@ -133,7 +132,7 @@ func (c *ColumnDefinitionEntity) ColumnDiff(
if c.columnDefinition.Type.Options.Collate = t1cc.collate; c.columnDefinition.Type.Options.Collate == "" {
collation := env.CollationEnv().DefaultCollationForCharset(t1cc.charset)
if collation == collations.Unknown {
return nil, vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "cannot match collation to charset %v", t1cc.charset)
return nil, &UnknownColumnCharsetCollationError{Column: c.columnDefinition.Name.String(), Charset: t1cc.charset}
}
c.columnDefinition.Type.Options.Collate = env.CollationEnv().LookupName(collation)
}
Expand All @@ -143,7 +142,7 @@ func (c *ColumnDefinitionEntity) ColumnDiff(
collationID := env.CollationEnv().LookupByName(other.columnDefinition.Type.Options.Collate)
charset := env.CollationEnv().LookupCharsetName(collationID)
if charset == "" {
return nil, vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "cannot match charset to collation %v", other.columnDefinition.Type.Options.Collate)
return nil, &UnknownColumnCollationCharsetError{Column: other.columnDefinition.Name.String(), Collation: other.columnDefinition.Type.Options.Collate}
}
defer func() {
other.columnDefinition.Type.Charset.Name = ""
Expand All @@ -160,7 +159,7 @@ func (c *ColumnDefinitionEntity) ColumnDiff(
if other.columnDefinition.Type.Options.Collate = t2cc.collate; other.columnDefinition.Type.Options.Collate == "" {
collation := env.CollationEnv().DefaultCollationForCharset(t2cc.charset)
if collation == collations.Unknown {
return nil, vterrors.Errorf(vtrpcpb.Code_INVALID_ARGUMENT, "cannot match collation to charset %v", t2cc.charset)
return nil, &UnknownColumnCharsetCollationError{Column: other.columnDefinition.Name.String(), Charset: t2cc.charset}
}
other.columnDefinition.Type.Options.Collate = env.CollationEnv().LookupName(collation)
}
Expand All @@ -171,6 +170,24 @@ func (c *ColumnDefinitionEntity) ColumnDiff(
return nil, nil
}

getEnumValuesMap := func(enumValues []string) map[string]int {
m := make(map[string]int, len(enumValues))
for i, enumValue := range enumValues {
m[enumValue] = i
}
return m
}
switch hints.EnumReorderStrategy {
case EnumReorderStrategyReject:
otherEnumValuesMap := getEnumValuesMap(other.columnDefinition.Type.EnumValues)
for ordinal, enumValue := range c.columnDefinition.Type.EnumValues {
if otherOrdinal, ok := otherEnumValuesMap[enumValue]; ok {
if ordinal != otherOrdinal {
return nil, &EnumValueOrdinalChangedError{Table: tableName, Column: c.columnDefinition.Name.String(), Value: enumValue, Ordinal: ordinal, NewOrdinal: otherOrdinal}
}
}
}
}
return NewModifyColumnDiffByDefinition(other.columnDefinition), nil
}

Expand Down
39 changes: 24 additions & 15 deletions go/vt/schemadiff/diff_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -32,17 +32,17 @@ func TestDiffTables(t *testing.T) {
env57, err := vtenv.New(vtenv.Options{MySQLServerVersion: "5.7.9"})
require.NoError(t, err)
tt := []struct {
name string
from string
to string
diff string
cdiff string
fromName string
toName string
action string
isError bool
hints *DiffHints
env *Environment
name string
from string
to string
diff string
cdiff string
fromName string
toName string
action string
expectError string
hints *DiffHints
env *Environment
}{
{
name: "identical",
Expand Down Expand Up @@ -220,7 +220,16 @@ func TestDiffTables(t *testing.T) {
AlterTableAlgorithmStrategy: AlterTableAlgorithmStrategyCopy,
TableCharsetCollateStrategy: TableCharsetCollateIgnoreAlways,
},
isError: true,
expectError: (&UnknownColumnCollationCharsetError{Column: "a", Collation: "latin1_nonexisting"}).Error(),
},
{
name: "error on unknown charset",
from: "create table t (a varchar(64)) default charset=latin_nonexisting collate=''",
to: "create table t (a varchar(64) CHARACTER SET latin1 COLLATE latin1_bin)",
hints: &DiffHints{
AlterTableAlgorithmStrategy: AlterTableAlgorithmStrategyCopy,
},
expectError: (&UnknownColumnCharsetCollationError{Column: "a", Charset: "latin_nonexisting"}).Error(),
},
{
name: "changing table level defaults with column specific settings",
Expand Down Expand Up @@ -335,9 +344,9 @@ func TestDiffTables(t *testing.T) {
dq, dqerr := DiffCreateTablesQueries(env, ts.from, ts.to, hints)
d, err := DiffTables(env, fromCreateTable, toCreateTable, hints)
switch {
case ts.isError:
assert.Error(t, err)
assert.Error(t, dqerr)
case ts.expectError != "":
assert.ErrorContains(t, err, ts.expectError)
assert.ErrorContains(t, dqerr, ts.expectError)
case ts.diff == "":
assert.NoError(t, err)
assert.NoError(t, dqerr)
Expand Down
30 changes: 30 additions & 0 deletions go/vt/schemadiff/errors.go
Original file line number Diff line number Diff line change
Expand Up @@ -446,3 +446,33 @@ type EntityNotFoundError struct {
func (e *EntityNotFoundError) Error() string {
return fmt.Sprintf("entity %s not found", sqlescape.EscapeID(e.Name))
}

type EnumValueOrdinalChangedError struct {
Table string
Column string
Value string
Ordinal int
NewOrdinal int
}

func (e *EnumValueOrdinalChangedError) Error() string {
return fmt.Sprintf("ordinal of %s changed in enum or set column %s.%s, from %d to %d", e.Value, sqlescape.EscapeID(e.Table), sqlescape.EscapeID(e.Column), e.Ordinal, e.NewOrdinal)
}

type UnknownColumnCharsetCollationError struct {
Column string
Charset string
}

func (e *UnknownColumnCharsetCollationError) Error() string {
return fmt.Sprintf("unable to determine collation for column %s with charset %q", sqlescape.EscapeID(e.Column), e.Charset)
}

type UnknownColumnCollationCharsetError struct {
Column string
Collation string
}

func (e *UnknownColumnCollationCharsetError) Error() string {
return fmt.Sprintf("unable to determine charset for column %s with collation %q", sqlescape.EscapeID(e.Column), e.Collation)
}
4 changes: 1 addition & 3 deletions go/vt/schemadiff/schema.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,9 +22,7 @@ import (
"strings"

"vitess.io/vitess/go/mysql/capabilities"
"vitess.io/vitess/go/vt/proto/vtrpc"
"vitess.io/vitess/go/vt/sqlparser"
"vitess.io/vitess/go/vt/vterrors"
"vitess.io/vitess/go/vt/vtgate/semantics"
)

Expand Down Expand Up @@ -1093,7 +1091,7 @@ func (s *Schema) getEntityColumnNames(entityName string, schemaInformation *decl
case *CreateViewEntity:
return s.getViewColumnNames(entity, schemaInformation)
}
return nil, vterrors.Errorf(vtrpc.Code_INTERNAL, "unexpected entity type for %v", entityName)
return nil, &UnsupportedEntityError{Entity: entity.Name(), Statement: entity.Create().CanonicalStatementString()}
}

// getTableColumnNames returns the names of columns in given table.
Expand Down
2 changes: 1 addition & 1 deletion go/vt/schemadiff/table.go
Original file line number Diff line number Diff line change
Expand Up @@ -1599,7 +1599,7 @@ func (c *CreateTableEntity) diffColumns(alterTable *sqlparser.AlterTable,
t2ColEntity := NewColumnDefinitionEntity(t2Col)

// check diff between before/after columns:
modifyColumnDiff, err := t1ColEntity.ColumnDiff(c.Env, t2ColEntity, t1cc, t2cc)
modifyColumnDiff, err := t1ColEntity.ColumnDiff(c.Env, c.Name(), t2ColEntity, t1cc, t2cc, hints)
if err != nil {
return err
}
Expand Down
Loading

0 comments on commit e9c513e

Please sign in to comment.