1
0

host_test.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384
  1. package database
  2. import (
  3. "os"
  4. "path/filepath"
  5. "strings"
  6. "testing"
  7. "github.com/mhsanaei/3x-ui/v3/internal/database/model"
  8. )
  9. // hostColumns is the set of columns initModels must create for the hosts table.
  10. func hostColumns() []string {
  11. return []string{
  12. "id", "inbound_id", "sort_order", "remark", "server_description", "is_disabled", "is_hidden", "tags",
  13. "address", "port",
  14. "security", "sni", "host_header", "path", "alpn", "fingerprint",
  15. "override_sni_from_address", "keep_sni_blank", "pinned_peer_cert_sha256",
  16. "verify_peer_cert_by_name", "allow_insecure", "ech_config_list",
  17. "mux_params", "sockopt_params", "final_mask", "vless_route",
  18. "exclude_from_sub_types", "mihomo_ip_version", "mihomo_x25519", "shuffle_host", "node_guids",
  19. "created_at", "updated_at",
  20. }
  21. }
  22. func assertHostSchema(t *testing.T) {
  23. t.Helper()
  24. m := GetDB().Migrator()
  25. if !m.HasTable("hosts") {
  26. t.Fatalf("hosts table not created by initModels")
  27. }
  28. for _, col := range hostColumns() {
  29. if !m.HasColumn(&model.Host{}, col) {
  30. t.Fatalf("hosts table missing column %q", col)
  31. }
  32. }
  33. }
  34. // TestHostAutoMigrateCreatesColumns verifies the hosts table and every expected
  35. // column exist after initModels (SQLite).
  36. func TestHostAutoMigrateCreatesColumns(t *testing.T) {
  37. if err := InitDB(filepath.Join(t.TempDir(), "x-ui.db")); err != nil {
  38. t.Fatalf("InitDB: %v", err)
  39. }
  40. t.Cleanup(func() { _ = CloseDB() })
  41. assertHostSchema(t)
  42. }
  43. // TestHostAutoMigrateCreatesColumns_Postgres is the dual-driver counterpart.
  44. func TestHostAutoMigrateCreatesColumns_Postgres(t *testing.T) {
  45. if strings.TrimSpace(os.Getenv("XUI_DB_DSN")) == "" || os.Getenv("XUI_DB_TYPE") != "postgres" {
  46. t.Skip("set XUI_DB_TYPE=postgres and XUI_DB_DSN to run the postgres schema test")
  47. }
  48. if err := InitDB(""); err != nil {
  49. t.Fatalf("InitDB: %v", err)
  50. }
  51. t.Cleanup(func() { _ = CloseDB() })
  52. assertHostSchema(t)
  53. }
  54. // TestPruneOrphanedHosts verifies a host whose inbound_id has no matching inbound
  55. // is removed by the prune step.
  56. func TestPruneOrphanedHosts(t *testing.T) {
  57. if err := InitDB(filepath.Join(t.TempDir(), "x-ui.db")); err != nil {
  58. t.Fatalf("InitDB: %v", err)
  59. }
  60. t.Cleanup(func() { _ = CloseDB() })
  61. db := GetDB()
  62. orphan := &model.Host{InboundId: 99999, Remark: "orphan"}
  63. if err := db.Create(orphan).Error; err != nil {
  64. t.Fatalf("create orphan host: %v", err)
  65. }
  66. if err := pruneOrphanedHosts(); err != nil {
  67. t.Fatalf("pruneOrphanedHosts: %v", err)
  68. }
  69. var cnt int64
  70. if err := db.Model(&model.Host{}).Where("id = ?", orphan.Id).Count(&cnt).Error; err != nil {
  71. t.Fatalf("count: %v", err)
  72. }
  73. if cnt != 0 {
  74. t.Fatalf("orphan host not pruned, count=%d", cnt)
  75. }
  76. }