186 lines
5.6 KiB
Go
186 lines
5.6 KiB
Go
/*
|
|
Copyright 2022 GitHub Inc.
|
|
See https://github.com/github/gh-ost/blob/master/LICENSE
|
|
*/
|
|
|
|
package logic
|
|
|
|
import (
|
|
"strings"
|
|
"testing"
|
|
|
|
test "github.com/openark/golib/tests"
|
|
|
|
"github.com/github/gh-ost/go/base"
|
|
"github.com/github/gh-ost/go/binlog"
|
|
"github.com/github/gh-ost/go/sql"
|
|
)
|
|
|
|
func TestApplierGenerateSqlModeQuery(t *testing.T) {
|
|
migrationContext := base.NewMigrationContext()
|
|
applier := NewApplier(migrationContext)
|
|
|
|
{
|
|
test.S(t).ExpectEquals(
|
|
applier.generateSqlModeQuery(),
|
|
`sql_mode = CONCAT(@@session.sql_mode, ',NO_AUTO_VALUE_ON_ZERO,STRICT_ALL_TABLES')`,
|
|
)
|
|
}
|
|
{
|
|
migrationContext.SkipStrictMode = true
|
|
migrationContext.AllowZeroInDate = false
|
|
test.S(t).ExpectEquals(
|
|
applier.generateSqlModeQuery(),
|
|
`sql_mode = CONCAT(@@session.sql_mode, ',NO_AUTO_VALUE_ON_ZERO')`,
|
|
)
|
|
}
|
|
{
|
|
migrationContext.SkipStrictMode = false
|
|
migrationContext.AllowZeroInDate = true
|
|
test.S(t).ExpectEquals(
|
|
applier.generateSqlModeQuery(),
|
|
`sql_mode = REPLACE(REPLACE(CONCAT(@@session.sql_mode, ',NO_AUTO_VALUE_ON_ZERO,STRICT_ALL_TABLES'), 'NO_ZERO_IN_DATE', ''), 'NO_ZERO_DATE', '')`,
|
|
)
|
|
}
|
|
{
|
|
migrationContext.SkipStrictMode = true
|
|
migrationContext.AllowZeroInDate = true
|
|
test.S(t).ExpectEquals(
|
|
applier.generateSqlModeQuery(),
|
|
`sql_mode = REPLACE(REPLACE(CONCAT(@@session.sql_mode, ',NO_AUTO_VALUE_ON_ZERO'), 'NO_ZERO_IN_DATE', ''), 'NO_ZERO_DATE', '')`,
|
|
)
|
|
}
|
|
}
|
|
|
|
func TestApplierUpdateModifiesUniqueKeyColumns(t *testing.T) {
|
|
columns := sql.NewColumnList([]string{"id", "item_id"})
|
|
columnValues := sql.ToColumnValues([]interface{}{123456, 42})
|
|
|
|
migrationContext := base.NewMigrationContext()
|
|
migrationContext.OriginalTableColumns = columns
|
|
migrationContext.UniqueKey = &sql.UniqueKey{
|
|
Name: t.Name(),
|
|
Columns: *columns,
|
|
}
|
|
|
|
applier := NewApplier(migrationContext)
|
|
|
|
t.Run("unmodified", func(t *testing.T) {
|
|
modifiedColumn, isModified := applier.updateModifiesUniqueKeyColumns(&binlog.BinlogDMLEvent{
|
|
DatabaseName: "test",
|
|
DML: binlog.UpdateDML,
|
|
NewColumnValues: columnValues,
|
|
WhereColumnValues: columnValues,
|
|
})
|
|
test.S(t).ExpectEquals(modifiedColumn, "")
|
|
test.S(t).ExpectFalse(isModified)
|
|
})
|
|
|
|
t.Run("modified", func(t *testing.T) {
|
|
modifiedColumn, isModified := applier.updateModifiesUniqueKeyColumns(&binlog.BinlogDMLEvent{
|
|
DatabaseName: "test",
|
|
DML: binlog.UpdateDML,
|
|
NewColumnValues: sql.ToColumnValues([]interface{}{123456, 24}),
|
|
WhereColumnValues: columnValues,
|
|
})
|
|
test.S(t).ExpectEquals(modifiedColumn, "item_id")
|
|
test.S(t).ExpectTrue(isModified)
|
|
})
|
|
}
|
|
|
|
func TestApplierBuildDMLEventQuery(t *testing.T) {
|
|
columns := sql.NewColumnList([]string{"id", "item_id"})
|
|
columnValues := sql.ToColumnValues([]interface{}{123456, 42})
|
|
|
|
migrationContext := base.NewMigrationContext()
|
|
migrationContext.OriginalTableName = "test"
|
|
migrationContext.OriginalTableColumns = columns
|
|
migrationContext.SharedColumns = columns
|
|
migrationContext.MappedSharedColumns = columns
|
|
migrationContext.UniqueKey = &sql.UniqueKey{
|
|
Name: t.Name(),
|
|
Columns: *columns,
|
|
}
|
|
|
|
applier := NewApplier(migrationContext)
|
|
|
|
t.Run("delete", func(t *testing.T) {
|
|
binlogEvent := &binlog.BinlogDMLEvent{
|
|
DatabaseName: "test",
|
|
DML: binlog.DeleteDML,
|
|
WhereColumnValues: columnValues,
|
|
}
|
|
|
|
res := applier.buildDMLEventQuery(binlogEvent)
|
|
test.S(t).ExpectEquals(len(res), 1)
|
|
test.S(t).ExpectNil(res[0].err)
|
|
test.S(t).ExpectEquals(strings.TrimSpace(res[0].query),
|
|
`delete /* gh-ost `+"`test`.`_test_gho`"+` */
|
|
from
|
|
`+"`test`.`_test_gho`"+`
|
|
where
|
|
((`+"`id`"+` = ?) and (`+"`item_id`"+` = ?))`)
|
|
|
|
test.S(t).ExpectEquals(len(res[0].args), 2)
|
|
test.S(t).ExpectEquals(res[0].args[0], 123456)
|
|
test.S(t).ExpectEquals(res[0].args[1], 42)
|
|
})
|
|
|
|
t.Run("insert", func(t *testing.T) {
|
|
binlogEvent := &binlog.BinlogDMLEvent{
|
|
DatabaseName: "test",
|
|
DML: binlog.InsertDML,
|
|
NewColumnValues: columnValues,
|
|
}
|
|
res := applier.buildDMLEventQuery(binlogEvent)
|
|
test.S(t).ExpectEquals(len(res), 1)
|
|
test.S(t).ExpectNil(res[0].err)
|
|
test.S(t).ExpectEquals(strings.TrimSpace(res[0].query),
|
|
`replace /* gh-ost `+"`test`.`_test_gho`"+` */ into
|
|
`+"`test`.`_test_gho`"+`
|
|
`+"(`id`, `item_id`)"+`
|
|
values
|
|
(?, ?)`)
|
|
test.S(t).ExpectEquals(len(res[0].args), 2)
|
|
test.S(t).ExpectEquals(res[0].args[0], 123456)
|
|
test.S(t).ExpectEquals(res[0].args[1], 42)
|
|
})
|
|
|
|
t.Run("update", func(t *testing.T) {
|
|
binlogEvent := &binlog.BinlogDMLEvent{
|
|
DatabaseName: "test",
|
|
DML: binlog.UpdateDML,
|
|
NewColumnValues: columnValues,
|
|
WhereColumnValues: columnValues,
|
|
}
|
|
res := applier.buildDMLEventQuery(binlogEvent)
|
|
test.S(t).ExpectEquals(len(res), 1)
|
|
test.S(t).ExpectNil(res[0].err)
|
|
test.S(t).ExpectEquals(strings.TrimSpace(res[0].query),
|
|
`update /* gh-ost `+"`test`.`_test_gho`"+` */
|
|
`+"`test`.`_test_gho`"+`
|
|
set
|
|
`+"`id`"+`=?, `+"`item_id`"+`=?
|
|
where
|
|
((`+"`id`"+` = ?) and (`+"`item_id`"+` = ?))`)
|
|
test.S(t).ExpectEquals(len(res[0].args), 4)
|
|
test.S(t).ExpectEquals(res[0].args[0], 123456)
|
|
test.S(t).ExpectEquals(res[0].args[1], 42)
|
|
test.S(t).ExpectEquals(res[0].args[2], 123456)
|
|
test.S(t).ExpectEquals(res[0].args[3], 42)
|
|
})
|
|
}
|
|
|
|
func TestApplierInstantDDL(t *testing.T) {
|
|
migrationContext := base.NewMigrationContext()
|
|
migrationContext.DatabaseName = "test"
|
|
migrationContext.OriginalTableName = "mytable"
|
|
migrationContext.AlterStatementOptions = "ADD INDEX (foo)"
|
|
applier := NewApplier(migrationContext)
|
|
|
|
t.Run("instantDDLstmt", func(t *testing.T) {
|
|
stmt := applier.generateInstantDDLQuery()
|
|
test.S(t).ExpectEquals(stmt, "ALTER /* gh-ost */ TABLE `test`.`mytable` ADD INDEX (foo), ALGORITHM=INSTANT")
|
|
})
|
|
}
|