Files
MyGoNavi/internal/db/mysql_metadata_test.go
Syngnat 74a422a5e2 🐛 fix(sync): 修复 Oracle 同步连接与 MySQL 备份导出异常
- 分离 Oracle/OceanBase Oracle 同步连接 Service Name 与选中 schema

- 兼容旧同步请求中 database 被 schema 覆盖的情况

- 规范 MySQL/MariaDB SHOW CREATE TABLE 标识符引用

Refs #549

Refs #518
2026-06-11 10:24:48 +08:00

135 lines
3.6 KiB
Go

package db
import (
"errors"
"reflect"
"testing"
)
func TestCollectMySQLDatabaseNames_FallsBackToCurrentDatabase(t *testing.T) {
t.Parallel()
got, err := collectMySQLDatabaseNames(func(query string) ([]map[string]interface{}, []string, error) {
switch query {
case mysqlDatabaseQueries[0]:
return nil, nil, errors.New("Error 1227 (42000): Access denied; you need (at least one of) the SHOW DATABASES privilege(s) for this operation")
case mysqlDatabaseQueries[1]:
return []map[string]interface{}{
{"Database": "biz_app"},
}, nil, nil
default:
return nil, nil, errors.New("unexpected query")
}
})
if err != nil {
t.Fatalf("collectMySQLDatabaseNames 返回错误: %v", err)
}
want := []string{"biz_app"}
if !reflect.DeepEqual(got, want) {
t.Fatalf("unexpected database names, got=%v want=%v", got, want)
}
}
func TestCollectMySQLDatabaseNames_PrefersShowDatabasesWhenAvailable(t *testing.T) {
t.Parallel()
got, err := collectMySQLDatabaseNames(func(query string) ([]map[string]interface{}, []string, error) {
switch query {
case mysqlDatabaseQueries[0]:
return []map[string]interface{}{
{"Database": "analytics"},
{"database": "audit"},
}, nil, nil
case mysqlDatabaseQueries[1]:
return []map[string]interface{}{
{"Database": "should_not_be_used"},
}, nil, nil
default:
return nil, nil, errors.New("unexpected query")
}
})
if err != nil {
t.Fatalf("collectMySQLDatabaseNames 返回错误: %v", err)
}
want := []string{"analytics", "audit"}
if !reflect.DeepEqual(got, want) {
t.Fatalf("unexpected database names, got=%v want=%v", got, want)
}
}
func TestCollectMySQLDatabaseNames_ReturnsOriginalErrorWhenNoDatabaseResolved(t *testing.T) {
t.Parallel()
expectErr := errors.New("show databases denied")
got, err := collectMySQLDatabaseNames(func(query string) ([]map[string]interface{}, []string, error) {
switch query {
case mysqlDatabaseQueries[0]:
return nil, nil, expectErr
case mysqlDatabaseQueries[1]:
return []map[string]interface{}{
{"Database": nil},
}, nil, nil
default:
return nil, nil, errors.New("unexpected query")
}
})
if err == nil {
t.Fatalf("期望返回错误,实际 got=%v", got)
}
if !errors.Is(err, expectErr) {
t.Fatalf("错误不符合预期: %v", err)
}
}
func TestBuildMySQLShowCreateTableQueryNormalizesQuotedIdentifiers(t *testing.T) {
t.Parallel()
tests := []struct {
name string
dbName string
tableName string
want string
}{
{
name: "plain db and quoted table",
dbName: "app",
tableName: `"activate_record"`,
want: "SHOW CREATE TABLE `app`.`activate_record`",
},
{
name: "escaped quoted qualified table overrides db",
dbName: "ignored",
tableName: `\"crm\".\"activate_record\"`,
want: "SHOW CREATE TABLE `crm`.`activate_record`",
},
{
name: "backtick escaping",
dbName: "app`prod",
tableName: "`audit``log`",
want: "SHOW CREATE TABLE `app``prod`.`audit``log`",
},
{
name: "quoted table containing dot is not split",
dbName: "app",
tableName: `"activate.record"`,
want: "SHOW CREATE TABLE `app`.`activate.record`",
},
{
name: "mixed quote artifact from UI row value",
dbName: "app",
tableName: `'activate_record"`,
want: "SHOW CREATE TABLE `app`.`activate_record`",
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := buildMySQLShowCreateTableQuery(tt.dbName, tt.tableName); got != tt.want {
t.Fatalf("buildMySQLShowCreateTableQuery(%q,%q)=%q,want=%q", tt.dbName, tt.tableName, got, tt.want)
}
})
}
}