| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697 |
- package sub
- import (
- "fmt"
- "path/filepath"
- "strings"
- "testing"
- "github.com/mhsanaei/3x-ui/v3/internal/database"
- "github.com/mhsanaei/3x-ui/v3/internal/database/model"
- )
- // TestGetSubs_DuplicateSettingsClients_Deduped reproduces #5134: multi-node
- // sync/import drift can leave the same client twice inside an inbound's
- // legacy settings.clients JSON while the normalized client_inbounds table
- // stays clean. The subscription output must still contain one profile per
- // (inbound, client).
- func TestGetSubs_DuplicateSettingsClients_Deduped(t *testing.T) {
- dbDir := t.TempDir()
- t.Setenv("XUI_DB_FOLDER", dbDir)
- if err := database.InitDB(filepath.Join(dbDir, "x-ui.db")); err != nil {
- t.Fatalf("InitDB: %v", err)
- }
- t.Cleanup(func() { _ = database.CloseDB() })
- const subId = "sub-dup"
- const email = "[email protected]"
- const uuid = "f1b9265f-26a8-4b75-9be2-c64a94b15de1"
- db := database.GetDB()
- settings := fmt.Sprintf(`{"clients": [
- {"id": %q, "email": %q, "subId": %q, "enable": true},
- {"id": %q, "email": %q, "subId": %q, "enable": true}
- ]}`, uuid, email, subId, uuid, email, subId)
- ib := &model.Inbound{
- UserId: 1,
- Tag: "dup-in",
- Enable: true,
- Port: 42001,
- Protocol: model.VLESS,
- Settings: settings,
- StreamSettings: `{"network": "tcp", "security": "none"}`,
- }
- if err := db.Create(ib).Error; err != nil {
- t.Fatalf("seed inbound: %v", err)
- }
- client := &model.ClientRecord{Email: email, SubID: subId, UUID: uuid, Enable: true}
- if err := db.Create(client).Error; err != nil {
- t.Fatalf("seed client: %v", err)
- }
- if err := db.Create(&model.ClientInbound{ClientId: client.Id, InboundId: ib.Id}).Error; err != nil {
- t.Fatalf("seed client_inbound: %v", err)
- }
- s := NewSubService(false, "-ieo")
- links, emails, _, _, err := s.GetSubs(subId, "sub.example.com")
- if err != nil {
- t.Fatalf("GetSubs: %v", err)
- }
- if len(links) != 1 {
- t.Fatalf("links = %d, want 1 (duplicate settings.clients entries must collapse)", len(links))
- }
- if len(emails) != 1 {
- t.Fatalf("emails = %d, want 1, got %v", len(emails), emails)
- }
- // Identity, not just count: the single surviving link must be for this client.
- if !strings.Contains(links[0], uuid) {
- t.Fatalf("surviving link must carry the client uuid %q, got %q", uuid, links[0])
- }
- }
- // TestMatchingClients_DedupsCaseInsensitiveEmail pins the dedup KEY, not just the count:
- // the two entries differ only by email case, so dropping strings.ToLower (or keying on
- // another field) yields two clients. The byte-identical dupes above can't catch that.
- func TestMatchingClients_DedupsCaseInsensitiveEmail(t *testing.T) {
- const subId = "s1"
- const uuid = "11111111-2222-4333-8444-555555555555"
- ib := &model.Inbound{
- Protocol: model.VLESS,
- Settings: `{"clients":[
- {"id":"` + uuid + `","email":"[email protected]","subId":"` + subId + `","enable":true},
- {"id":"` + uuid + `","email":"[email protected]","subId":"` + subId + `","enable":true}
- ]}`,
- }
- s := &SubService{}
- got := s.matchingClients(ib, subId)
- if len(got) != 1 {
- t.Fatalf("case-differing duplicate emails must dedup to 1 client, got %d", len(got))
- }
- if got[0].Email != "[email protected]" {
- t.Fatalf("first occurrence must be kept, got %q", got[0].Email)
- }
- // A wrong subId must still be excluded (guards the subId filter at service.go:127).
- if other := s.matchingClients(ib, "nope"); len(other) != 0 {
- t.Fatalf("non-matching subId must yield 0 clients, got %d", len(other))
- }
- }
|