package sync import ( "GoNavi-Wails/internal/connection" "testing" ) func TestQuoteQualifiedIdentByType_KingbaseLeavesLowercaseQualifiedTableUnquoted(t *testing.T) { t.Parallel() got := quoteQualifiedIdentByType("kingbase", "ldf_server.andon_events") if got != "ldf_server.andon_events" { t.Fatalf("unexpected kingbase qualified identifier: %s", got) } } func TestQuoteQualifiedIdentByType_KingbaseNormalizesEscapedQuotedQualifiedTable(t *testing.T) { t.Parallel() got := quoteQualifiedIdentByType("kingbase", `\"Idf_server\".\"andon_events\"`) if got != `"Idf_server".andon_events` { t.Fatalf("unexpected kingbase qualified identifier: %s", got) } } func TestQuoteQualifiedIdentByType_KingbaseAliasUsesKingbaseQuoting(t *testing.T) { t.Parallel() got := quoteQualifiedIdentByType("kingbase8", `\"ldf_server\".\"andon_events\"`) if got != "ldf_server.andon_events" { t.Fatalf("unexpected kingbase alias qualified identifier: %s", got) } } func TestQuoteIdentByType_KingbaseStillQuotesReservedAndMixedCaseIdentifiers(t *testing.T) { t.Parallel() if got := quoteIdentByType("kingbase", "select"); got != `"select"` { t.Fatalf("expected reserved word to stay quoted, got %s", got) } if got := quoteIdentByType("kingbase", "CamelName"); got != `"CamelName"` { t.Fatalf("expected mixed-case identifier to stay quoted, got %s", got) } } func TestNormalizeSchemaAndTable_KingbaseNormalizesEscapedQualifiedName(t *testing.T) { t.Parallel() schema, table := normalizeSchemaAndTable("kingbase", "demo", `\"Idf_server\".\"andon_events\"`) if schema != "Idf_server" || table != "andon_events" { t.Fatalf("unexpected kingbase schema/table: %q.%q", schema, table) } } func TestNormalizeSyncTargetSchemaAndTable_UsesExplicitSchemaForPGLikeTargets(t *testing.T) { t.Parallel() config := SyncConfig{ TargetConfig: connection.ConnectionConfig{Type: "opengauss", Database: "analytics"}, TargetDatabase: "analytics", TargetSchema: "reporting", } schema, table := normalizeSyncTargetSchemaAndTable(config, "orders") if schema != "reporting" || table != "orders" { t.Fatalf("unexpected opengauss target schema/table: %q.%q", schema, table) } } func TestNormalizeSyncTargetSchemaAndTable_UsesTargetSchemaWhenSourceTableIsQualified(t *testing.T) { t.Parallel() config := SyncConfig{ TargetConfig: connection.ConnectionConfig{Type: "opengauss", Database: "analytics"}, TargetDatabase: "analytics", TargetSchema: "reporting", } schema, table := normalizeSyncTargetSchemaAndTable(config, "archive.orders") if schema != "reporting" || table != "orders" { t.Fatalf("unexpected qualified source table override result: %q.%q", schema, table) } } func TestNormalizeSyncTargetSchemaAndTable_KeepsQualifiedTargetTableInSourceQueryMode(t *testing.T) { t.Parallel() config := SyncConfig{ TargetConfig: connection.ConnectionConfig{Type: "opengauss", Database: "analytics"}, TargetDatabase: "analytics", TargetSchema: "reporting", SourceQuery: "select * from archive.orders", } schema, table := normalizeSyncTargetSchemaAndTable(config, "archive.orders") if schema != "archive" || table != "orders" { t.Fatalf("unexpected qualified target schema/table: %q.%q", schema, table) } } func TestNormalizeSyncTargetSchemaAndTable_UsesQualifiedTableForSQLServerExplicitSchema(t *testing.T) { t.Parallel() config := SyncConfig{ TargetConfig: connection.ConnectionConfig{Type: "sqlserver", Database: "warehouse"}, TargetDatabase: "warehouse", TargetSchema: "sales", } schema, table := normalizeSyncTargetSchemaAndTable(config, "archive.orders") if schema != "warehouse" || table != "sales.orders" { t.Fatalf("unexpected sqlserver target schema/table: %q.%q", schema, table) } } func TestNormalizeSyncTargetSchemaAndTable_UsesQualifiedTableForDuckDBExplicitSchema(t *testing.T) { t.Parallel() config := SyncConfig{ TargetConfig: connection.ConnectionConfig{Type: "duckdb", Database: "analytics"}, TargetDatabase: "analytics", TargetSchema: "reporting", } schema, table := normalizeSyncTargetSchemaAndTable(config, "orders") if schema != "analytics" || table != "reporting.orders" { t.Fatalf("unexpected duckdb target schema/table: %q.%q", schema, table) } } func TestNormalizeSyncConnectionDatabasesKeepsOracleServiceName(t *testing.T) { t.Parallel() config := SyncConfig{ SourceConfig: connection.ConnectionConfig{Type: "oracle", Database: "ORCLPDB1"}, SourceDatabase: "APP_SCHEMA", TargetConfig: connection.ConnectionConfig{Type: "mysql", Database: "old_target"}, TargetDatabase: "warehouse", } got := normalizeSyncConnectionDatabases(config) if got.SourceConfig.Database != "ORCLPDB1" { t.Fatalf("Oracle 连接 Service Name 不应被 schema 覆盖,got=%q", got.SourceConfig.Database) } if selectedSyncSourceDatabase(got) != "APP_SCHEMA" { t.Fatalf("Oracle 选中 schema 应保留在 SourceDatabase,got=%q", selectedSyncSourceDatabase(got)) } if got.TargetConfig.Database != "warehouse" { t.Fatalf("非 Oracle 目标库应继续写入连接 Database,got=%q", got.TargetConfig.Database) } } func TestNormalizeSyncConnectionDatabasesKeepsOceanBaseOracleServiceName(t *testing.T) { t.Parallel() config := SyncConfig{ SourceConfig: connection.ConnectionConfig{ Type: "oceanbase", Database: "ORCL", ConnectionParams: "protocol=oracle", }, SourceDatabase: "APP_SCHEMA", } got := normalizeSyncConnectionDatabases(config) if got.SourceConfig.Database != "ORCL" { t.Fatalf("OceanBase Oracle 服务名不应被 schema 覆盖,got=%q", got.SourceConfig.Database) } if selectedSyncSourceDatabase(got) != "APP_SCHEMA" { t.Fatalf("OceanBase Oracle 选中 schema 应保留在 SourceDatabase,got=%q", selectedSyncSourceDatabase(got)) } } func TestNormalizeMigrationDBType_KingbaseAliases(t *testing.T) { t.Parallel() for _, in := range []string{"kingbase8", "kingbasees", "kingbasev8"} { if got := normalizeMigrationDBType(in); got != "kingbase" { t.Fatalf("normalizeMigrationDBType(%q)=%q, want kingbase", in, got) } } } func TestNormalizeMigrationDBType_SQLServerAndIRISAliases(t *testing.T) { t.Parallel() if got := normalizeMigrationDBType("mssql"); got != "sqlserver" { t.Fatalf("normalizeMigrationDBType(%q)=%q, want sqlserver", "mssql", got) } if got := resolveMigrationDBType(connection.ConnectionConfig{Type: "custom", Driver: "inter-systems-iris"}); got != "iris" { t.Fatalf("resolveMigrationDBType(custom iris)=%q, want iris", got) } } func TestQualifiedNameForQuery_UsesSchemaAwareTargets(t *testing.T) { t.Parallel() if got := qualifiedNameForQuery("sqlserver", "warehouse", "orders", "orders"); got != "dbo.orders" { t.Fatalf("unexpected sqlserver qualified name: %s", got) } if got := qualifiedNameForQuery("sqlserver", "warehouse", "sales.orders", "orders"); got != "sales.orders" { t.Fatalf("unexpected sqlserver explicit qualified name: %s", got) } if got := qualifiedNameForQuery("oracle", "APP_SCHEMA", "orders", "orders"); got != "APP_SCHEMA.orders" { t.Fatalf("unexpected oracle qualified name: %s", got) } if got := qualifiedNameForQuery("dameng", "APP_SCHEMA", "orders", "orders"); got != "APP_SCHEMA.orders" { t.Fatalf("unexpected dameng qualified name: %s", got) } if got := qualifiedNameForQuery("iris", "APP_SCHEMA", "orders", "orders"); got != "APP_SCHEMA.orders" { t.Fatalf("unexpected iris qualified name: %s", got) } if got := qualifiedNameForQuery("duckdb", "analytics", "reporting.orders", "orders"); got != "reporting.orders" { t.Fatalf("unexpected duckdb qualified name: %s", got) } } func TestShouldUseQualifiedSyncApplyTable(t *testing.T) { t.Parallel() if !shouldUseQualifiedSyncApplyTable(connection.ConnectionConfig{Type: "oracle"}) { t.Fatal("oracle should apply against qualified target table") } if !shouldUseQualifiedSyncApplyTable(connection.ConnectionConfig{Type: "duckdb"}) { t.Fatal("duckdb should apply against qualified target table") } if !shouldUseQualifiedSyncApplyTable(connection.ConnectionConfig{Type: "oceanbase", OceanBaseProtocol: "oracle"}) { t.Fatal("oceanbase oracle should apply against qualified target table") } if shouldUseQualifiedSyncApplyTable(connection.ConnectionConfig{Type: "mysql"}) { t.Fatal("mysql should keep raw target table for apply") } } func TestBuildPagedSourceTableQuery_MySQLUsesStablePKPagination(t *testing.T) { t.Parallel() query := buildPagedSourceTableQuery("mysql", "app.events", []connection.ColumnDefinition{ {Name: "id"}, {Name: "name"}, }, "id", 1000, 2000) want := "SELECT `id`, `name` FROM `app`.`events` ORDER BY `id` ASC LIMIT 1000 OFFSET 2000" if query != want { t.Fatalf("unexpected paged query:\n got: %s\nwant: %s", query, want) } } func TestBuildPagedSourceTableQuery_SQLServerUsesOuterAliasColumns(t *testing.T) { t.Parallel() query := buildPagedSourceTableQuery("sqlserver", "dbo.events", []connection.ColumnDefinition{ {Name: "id"}, {Name: "name"}, }, "id", 1000, 2000) want := "SELECT [__gonavi_page_result__].[id], [__gonavi_page_result__].[name] FROM (SELECT [__gonavi_page__].*, ROW_NUMBER() OVER (ORDER BY [id] ASC) AS [__gonavi_rn__] FROM (SELECT [id], [name] FROM [dbo].[events]) AS [__gonavi_page__]) AS [__gonavi_page_result__] WHERE [__gonavi_rn__] > 2000 AND [__gonavi_rn__] <= 3000 ORDER BY [__gonavi_rn__]" if query != want { t.Fatalf("unexpected paged query:\n got: %s\nwant: %s", query, want) } } func TestIsSamePhysicalSyncTableDetectsFullOverwriteSelfTarget(t *testing.T) { t.Parallel() cfg := SyncConfig{ SourceConfig: connection.ConnectionConfig{Type: "mysql", Host: "127.0.0.1", Port: 3306, Database: "app"}, TargetConfig: connection.ConnectionConfig{Type: "mysql", Host: "127.0.0.1", Port: 3306, Database: "app"}, } plan := SchemaMigrationPlan{SourceQueryTable: "app.events", TargetQueryTable: "app.events"} if !isSamePhysicalSyncTable(cfg, plan, "mysql", "mysql") { t.Fatal("expected identical connection/table to be detected") } cfg.TargetConfig.Database = "archive" if isSamePhysicalSyncTable(cfg, plan, "mysql", "mysql") { t.Fatal("different database should not be treated as same physical table") } } func TestBuildPKInSelectQueryEscapesStringLiterals(t *testing.T) { t.Parallel() query := buildPKInSelectQuery("mysql", "app.users", []connection.ColumnDefinition{ {Name: "id"}, {Name: "name"}, }, "id", []interface{}{"a'1", "b2"}) want := "SELECT `id`, `name` FROM `app`.`users` WHERE `id` IN ('a''1', 'b2')" if query != want { t.Fatalf("unexpected PK IN query:\n got: %s\nwant: %s", query, want) } } func TestBuildKeysetPagedTableQueryUsesLastPK(t *testing.T) { t.Parallel() query := buildKeysetPagedTableQuery("mysql", "app.users", []connection.ColumnDefinition{{Name: "id"}}, "id", 100, true, 50) want := "SELECT `id` FROM `app`.`users` WHERE `id` > 100 ORDER BY `id` ASC LIMIT 50" if query != want { t.Fatalf("unexpected keyset query:\n got: %s\nwant: %s", query, want) } } func TestBuildSourceQueryPageSQLWrapsSelect(t *testing.T) { t.Parallel() query := buildSourceQueryPageSQL("mysql", "SELECT id, name FROM active_users;", "id", 1000, 2000) want := "SELECT * FROM (SELECT id, name FROM active_users) AS __gonavi_source_query__ ORDER BY `id` ASC LIMIT 1000 OFFSET 2000" if query != want { t.Fatalf("unexpected source query page SQL:\n got: %s\nwant: %s", query, want) } } func TestNormalizeSourceQueryForPagingRejectsMultiStatement(t *testing.T) { t.Parallel() if _, ok := normalizeSourceQueryForPaging("SELECT * FROM users; DELETE FROM users"); ok { t.Fatal("expected multi-statement source query to be rejected for pagination") } }