migrate_data_test.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139
  1. package database
  2. import (
  3. "os"
  4. "testing"
  5. "github.com/mhsanaei/3x-ui/v3/database/model"
  6. "gorm.io/driver/postgres"
  7. "gorm.io/driver/sqlite"
  8. "gorm.io/gorm"
  9. "gorm.io/gorm/logger"
  10. )
  11. func TestMigrateData_CompositeKeyTableLargerThanBatch(t *testing.T) {
  12. dsn := os.Getenv("XUI_TEST_PG_DSN")
  13. if dsn == "" {
  14. t.Skip("set XUI_TEST_PG_DSN to a reachable Postgres to run this test")
  15. }
  16. // Seed a SQLite source with the full schema and >500 client_inbounds rows.
  17. srcPath := t.TempDir() + "/x-ui.db"
  18. src, err := gorm.Open(sqlite.Open(srcPath), &gorm.Config{Logger: logger.Discard})
  19. if err != nil {
  20. t.Fatalf("open sqlite: %v", err)
  21. }
  22. for _, m := range migrationModels() {
  23. if err := src.AutoMigrate(m); err != nil {
  24. t.Fatalf("automigrate %T: %v", m, err)
  25. }
  26. }
  27. const n = 600 // > batchSize (500) so the between-batches path is exercised
  28. links := make([]model.ClientInbound, 0, n)
  29. for i := 1; i <= n; i++ {
  30. links = append(links, model.ClientInbound{ClientId: i, InboundId: 1})
  31. }
  32. if err := src.CreateInBatches(links, 200).Error; err != nil {
  33. t.Fatalf("seed client_inbounds: %v", err)
  34. }
  35. if sqlDB, err := src.DB(); err == nil {
  36. sqlDB.Close() // flush before MigrateData reopens the file
  37. }
  38. // Make the test re-runnable: drop any tables from a previous run.
  39. dst, err := gorm.Open(postgres.Open(dsn), &gorm.Config{Logger: logger.Discard})
  40. if err != nil {
  41. t.Fatalf("open postgres: %v", err)
  42. }
  43. if err := dst.Migrator().DropTable(migrationModels()...); err != nil {
  44. t.Fatalf("drop tables: %v", err)
  45. }
  46. if err := MigrateData(srcPath, dsn); err != nil {
  47. t.Fatalf("MigrateData: %v", err) // fails here before the fix
  48. }
  49. var got int64
  50. if err := dst.Model(&model.ClientInbound{}).Count(&got).Error; err != nil {
  51. t.Fatalf("count: %v", err)
  52. }
  53. if got != n {
  54. t.Fatalf("client_inbounds rows = %d, want %d", got, n)
  55. }
  56. }
  57. func TestMigrateData_PreservesFalseDefaultedColumns(t *testing.T) {
  58. dsn := os.Getenv("XUI_TEST_PG_DSN")
  59. if dsn == "" {
  60. t.Skip("set XUI_TEST_PG_DSN to a reachable Postgres to run this test")
  61. }
  62. srcPath := t.TempDir() + "/x-ui.db"
  63. src, err := gorm.Open(sqlite.Open(srcPath), &gorm.Config{Logger: logger.Discard})
  64. if err != nil {
  65. t.Fatalf("open sqlite: %v", err)
  66. }
  67. for _, m := range migrationModels() {
  68. if err := src.AutoMigrate(m); err != nil {
  69. t.Fatalf("automigrate %T: %v", m, err)
  70. }
  71. }
  72. if err := src.Create([]*model.ClientRecord{
  73. {Email: "[email protected]"},
  74. {Email: "[email protected]"},
  75. }).Error; err != nil {
  76. t.Fatalf("seed clients: %v", err)
  77. }
  78. if err := src.Model(&model.ClientRecord{}).Where("email = ?", "[email protected]").
  79. Update("enable", false).Error; err != nil {
  80. t.Fatalf("disable client: %v", err)
  81. }
  82. if err := src.Create(&model.Node{Name: "n-off", Address: "1.2.3.4", Port: 1, ApiToken: "tok"}).Error; err != nil {
  83. t.Fatalf("seed node: %v", err)
  84. }
  85. if err := src.Model(&model.Node{}).Where("name = ?", "n-off").
  86. Update("enable", false).Error; err != nil {
  87. t.Fatalf("disable node: %v", err)
  88. }
  89. if sqlDB, err := src.DB(); err == nil {
  90. sqlDB.Close()
  91. }
  92. dst, err := gorm.Open(postgres.Open(dsn), &gorm.Config{Logger: logger.Discard})
  93. if err != nil {
  94. t.Fatalf("open postgres: %v", err)
  95. }
  96. if err := dst.Migrator().DropTable(migrationModels()...); err != nil {
  97. t.Fatalf("drop tables: %v", err)
  98. }
  99. if err := MigrateData(srcPath, dsn); err != nil {
  100. t.Fatalf("MigrateData: %v", err)
  101. }
  102. var off model.ClientRecord
  103. if err := dst.Where("email = ?", "[email protected]").First(&off).Error; err != nil {
  104. t.Fatalf("load disabled client: %v", err)
  105. }
  106. if off.Enable {
  107. t.Fatalf("disabled client re-enabled after migration (enable=%v)", off.Enable)
  108. }
  109. var on model.ClientRecord
  110. if err := dst.Where("email = ?", "[email protected]").First(&on).Error; err != nil {
  111. t.Fatalf("load enabled client: %v", err)
  112. }
  113. if !on.Enable {
  114. t.Fatalf("enabled client wrongly disabled after migration")
  115. }
  116. var node model.Node
  117. if err := dst.Where("name = ?", "n-off").First(&node).Error; err != nil {
  118. t.Fatalf("load node: %v", err)
  119. }
  120. if node.Enable {
  121. t.Fatalf("disabled node re-enabled after migration")
  122. }
  123. }