mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-06-28 09:21:38 +08:00
252 lines
7.4 KiB
Go
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())
|
|
})
|
|
}
|
|
}
|