Files
MyGoNavi/internal/sync/migration_tdengine_target_i18n_test.go
2026-06-22 15:12:42 +08:00

252 lines
7.4 KiB
Go

package sync
import (
"GoNavi-Wails/internal/connection"
"GoNavi-Wails/shared/i18n"
"errors"
"os"
"strings"
"testing"
)
func baseMySQLLikeToTDengineI18nConfig() SyncConfig {
return SyncConfig{
SourceConfig: connection.ConnectionConfig{Type: "mysql", Database: "shop"},
TargetConfig: connection.ConnectionConfig{Type: "tdengine", Database: "taos"},
}
}
func basePGLikeToTDengineI18nConfig() SyncConfig {
return SyncConfig{
SourceConfig: connection.ConnectionConfig{Type: "postgres", Database: "ignored"},
TargetConfig: connection.ConnectionConfig{Type: "tdengine", Database: "taos"},
}
}
func baseClickHouseToTDengineI18nConfig() SyncConfig {
return SyncConfig{
SourceConfig: connection.ConnectionConfig{Type: "clickhouse", Database: "analytics"},
TargetConfig: connection.ConnectionConfig{Type: "tdengine", Database: "taos"},
}
}
func baseTDengineToTDengineI18nConfig() SyncConfig {
return SyncConfig{
SourceConfig: connection.ConnectionConfig{Type: "tdengine", Database: "metrics"},
TargetConfig: connection.ConnectionConfig{Type: "tdengine", Database: "archive"},
}
}
func assertNoLegacyTDengineTargetMigrationChinese(t *testing.T, text string) {
t.Helper()
legacyFragments := []string{
"\u83b7\u53d6\u6e90\u8868\u5b57\u6bb5\u5931\u8d25",
"\u6e90\u8868\u4e0d\u5b58\u5728\u6216\u65e0\u5217\u5b9a\u4e49",
"\u83b7\u53d6\u76ee\u6807\u8868\u5b57\u6bb5\u5931\u8d25",
}
for _, fragment := range legacyFragments {
if strings.Contains(text, fragment) {
t.Fatalf("expected no legacy TDengine target migration Chinese fragment %q in %q", fragment, text)
}
}
}
func TestTDengineTargetMigrationUsesLocalizedBackendTextSourceGuard(t *testing.T) {
sourceBytes, err := os.ReadFile("migration_tdengine_target.go")
if err != nil {
t.Fatalf("read migration_tdengine_target.go: %v", err)
}
source := string(sourceBytes)
legacyMessages := []string{
"fmt.Errorf(\"\u83b7\u53d6\u6e90\u8868\u5b57\u6bb5\u5931\u8d25: %w\", err)",
"fmt.Errorf(\"\u6e90\u8868\u4e0d\u5b58\u5728\u6216\u65e0\u5217\u5b9a\u4e49: %s\", tableName)",
"fmt.Errorf(\"\u83b7\u53d6\u76ee\u6807\u8868\u5b57\u6bb5\u5931\u8d25: %w\", err)",
}
for _, legacy := range legacyMessages {
if strings.Contains(source, legacy) {
t.Fatalf("migration_tdengine_target.go still contains legacy raw user-visible message %q", legacy)
}
}
requiredKeys := []string{
"data_sync.backend.error.source_table_columns_failed",
"data_sync.backend.error.source_table_missing_or_no_columns",
"data_sync.backend.error.target_table_columns_failed",
}
for _, key := range requiredKeys {
if !strings.Contains(source, key) {
t.Fatalf("migration_tdengine_target.go should reference localized key %q", key)
}
}
}
func TestTDengineTargetMigrationCatalogKeysExist(t *testing.T) {
catalogs, err := i18n.LoadCatalogs()
if err != nil {
t.Fatalf("LoadCatalogs() error = %v", err)
}
keys := []string{
"data_sync.backend.error.source_table_columns_failed",
"data_sync.backend.error.source_table_missing_or_no_columns",
"data_sync.backend.error.target_table_columns_failed",
}
for _, language := range i18n.SupportedLanguages() {
catalog := catalogs[language]
for _, key := range keys {
if strings.TrimSpace(catalog[key]) == "" {
t.Fatalf("%s catalog missing TDengine target migration key %q", language, key)
}
}
}
}
func TestTDengineTargetMigrationUsesCurrentLanguageForStructuredErrors(t *testing.T) {
SetBackendLanguage(i18n.LanguageEnUS)
t.Cleanup(func() {
SetBackendLanguage(i18n.LanguageZhCN)
})
tableName := "metrics"
sourceColumnsErr := errors.New("source columns boom")
targetColumnsErr := errors.New("target columns boom")
mysqlLikeSourceCols := []connection.ColumnDefinition{
{Name: "id", Type: "bigint", Nullable: "NO"},
{Name: "ts", Type: "datetime", Nullable: "NO"},
{Name: "payload", Type: "json", Nullable: "YES"},
}
clickHouseSourceCols := []connection.ColumnDefinition{
{Name: "event_time", Type: "DateTime64(3)", Nullable: "NO"},
{Name: "host", Type: "FixedString(64)", Nullable: "YES"},
}
tdengineSourceCols := []connection.ColumnDefinition{
{Name: "ts", Type: "TIMESTAMP", Nullable: "NO"},
{Name: "value", Type: "DOUBLE", Nullable: "YES"},
}
cases := []struct {
name string
run func() error
wantKey string
wantParams map[string]any
wantCause error
}{
{
name: "source table columns failed",
run: func() error {
_, _, _, err := buildMySQLLikeToTDenginePlan(
baseMySQLLikeToTDengineI18nConfig(),
tableName,
&errorMigrationDB{getColumnsErr: sourceColumnsErr},
&fakeMigrationDB{},
)
return err
},
wantKey: "data_sync.backend.error.source_table_columns_failed",
wantParams: map[string]any{
"detail": "source columns boom",
},
wantCause: sourceColumnsErr,
},
{
name: "source table missing or no columns",
run: func() error {
_, _, _, err := buildPGLikeToTDenginePlan(
basePGLikeToTDengineI18nConfig(),
tableName,
&fakeMigrationDB{},
&fakeMigrationDB{},
)
return err
},
wantKey: "data_sync.backend.error.source_table_missing_or_no_columns",
wantParams: map[string]any{
"table": tableName,
},
},
{
name: "target table columns failed mysqllike source",
run: func() error {
_, _, _, err := buildMySQLLikeToTDenginePlan(
baseMySQLLikeToTDengineI18nConfig(),
tableName,
&fakeMigrationDB{
columns: map[string][]connection.ColumnDefinition{
"shop.metrics": mysqlLikeSourceCols,
},
},
&errorMigrationDB{getColumnsErr: targetColumnsErr},
)
return err
},
wantKey: "data_sync.backend.error.target_table_columns_failed",
wantParams: map[string]any{
"detail": "target columns boom",
},
wantCause: targetColumnsErr,
},
{
name: "target table columns failed clickhouse source",
run: func() error {
_, _, _, err := buildClickHouseToTDenginePlan(
baseClickHouseToTDengineI18nConfig(),
tableName,
&fakeMigrationDB{
columns: map[string][]connection.ColumnDefinition{
"analytics.metrics": clickHouseSourceCols,
},
},
&errorMigrationDB{getColumnsErr: targetColumnsErr},
)
return err
},
wantKey: "data_sync.backend.error.target_table_columns_failed",
wantParams: map[string]any{
"detail": "target columns boom",
},
wantCause: targetColumnsErr,
},
{
name: "target table columns failed tdengine source",
run: func() error {
_, _, _, err := buildTDengineToTDenginePlan(
baseTDengineToTDengineI18nConfig(),
tableName,
&fakeMigrationDB{
columns: map[string][]connection.ColumnDefinition{
"metrics.metrics": tdengineSourceCols,
},
},
&errorMigrationDB{getColumnsErr: targetColumnsErr},
)
return err
},
wantKey: "data_sync.backend.error.target_table_columns_failed",
wantParams: map[string]any{
"detail": "target columns boom",
},
wantCause: targetColumnsErr,
},
}
for _, tc := range cases {
t.Run(tc.name, func(t *testing.T) {
err := tc.run()
if err == nil {
t.Fatalf("expected error for %s", tc.name)
}
want := localizedSyncTestText(t, i18n.LanguageEnUS, tc.wantKey, tc.wantParams)
if err.Error() != want {
t.Fatalf("expected localized TDengine target migration message %q, got %q", want, err.Error())
}
if tc.wantCause != nil && !errors.Is(err, tc.wantCause) {
t.Fatalf("expected TDengine target migration error to wrap %v, got %v", tc.wantCause, err)
}
assertNoLegacyTDengineTargetMigrationChinese(t, err.Error())
})
}
}