| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384 |
- package database
- import (
- "os"
- "path/filepath"
- "strings"
- "testing"
- "github.com/mhsanaei/3x-ui/v3/internal/database/model"
- )
- // hostColumns is the set of columns initModels must create for the hosts table.
- func hostColumns() []string {
- return []string{
- "id", "inbound_id", "sort_order", "remark", "server_description", "is_disabled", "is_hidden", "tags",
- "address", "port",
- "security", "sni", "host_header", "path", "alpn", "fingerprint",
- "override_sni_from_address", "keep_sni_blank", "pinned_peer_cert_sha256",
- "verify_peer_cert_by_name", "allow_insecure", "ech_config_list",
- "mux_params", "sockopt_params", "final_mask", "vless_route",
- "exclude_from_sub_types", "mihomo_ip_version", "mihomo_x25519", "shuffle_host", "node_guids",
- "created_at", "updated_at",
- }
- }
- func assertHostSchema(t *testing.T) {
- t.Helper()
- m := GetDB().Migrator()
- if !m.HasTable("hosts") {
- t.Fatalf("hosts table not created by initModels")
- }
- for _, col := range hostColumns() {
- if !m.HasColumn(&model.Host{}, col) {
- t.Fatalf("hosts table missing column %q", col)
- }
- }
- }
- // TestHostAutoMigrateCreatesColumns verifies the hosts table and every expected
- // column exist after initModels (SQLite).
- func TestHostAutoMigrateCreatesColumns(t *testing.T) {
- if err := InitDB(filepath.Join(t.TempDir(), "x-ui.db")); err != nil {
- t.Fatalf("InitDB: %v", err)
- }
- t.Cleanup(func() { _ = CloseDB() })
- assertHostSchema(t)
- }
- // TestHostAutoMigrateCreatesColumns_Postgres is the dual-driver counterpart.
- func TestHostAutoMigrateCreatesColumns_Postgres(t *testing.T) {
- if strings.TrimSpace(os.Getenv("XUI_DB_DSN")) == "" || os.Getenv("XUI_DB_TYPE") != "postgres" {
- t.Skip("set XUI_DB_TYPE=postgres and XUI_DB_DSN to run the postgres schema test")
- }
- if err := InitDB(""); err != nil {
- t.Fatalf("InitDB: %v", err)
- }
- t.Cleanup(func() { _ = CloseDB() })
- assertHostSchema(t)
- }
- // TestPruneOrphanedHosts verifies a host whose inbound_id has no matching inbound
- // is removed by the prune step.
- func TestPruneOrphanedHosts(t *testing.T) {
- if err := InitDB(filepath.Join(t.TempDir(), "x-ui.db")); err != nil {
- t.Fatalf("InitDB: %v", err)
- }
- t.Cleanup(func() { _ = CloseDB() })
- db := GetDB()
- orphan := &model.Host{InboundId: 99999, Remark: "orphan"}
- if err := db.Create(orphan).Error; err != nil {
- t.Fatalf("create orphan host: %v", err)
- }
- if err := pruneOrphanedHosts(); err != nil {
- t.Fatalf("pruneOrphanedHosts: %v", err)
- }
- var cnt int64
- if err := db.Model(&model.Host{}).Where("id = ?", orphan.Id).Count(&cnt).Error; err != nil {
- t.Fatalf("count: %v", err)
- }
- if cnt != 0 {
- t.Fatalf("orphan host not pruned, count=%d", cnt)
- }
- }
|