| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364 |
- package service
- import (
- "path/filepath"
- "testing"
- "github.com/mhsanaei/3x-ui/v3/internal/database"
- "github.com/mhsanaei/3x-ui/v3/internal/database/model"
- )
- // EffectiveFlowsByEmails resolves intended flow for many clients in one batched
- // query, taking the flow_override of the lowest inbound_id and skipping emails
- // with no non-empty flow anywhere.
- func TestEffectiveFlowsByEmails(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() })
- db := database.GetDB()
- const vision = "xtls-rprx-vision"
- // vis@x: attached to inbound 20 (empty flow) and 10 (Vision) — lowest
- // inbound_id (10) wins, so the empty override on 20 must not mask it.
- // plain@x: only an empty flow_override anywhere — absent from the result.
- mkClient := func(id int, email string) {
- if err := db.Create(&model.ClientRecord{Id: id, Email: email, Enable: true}).Error; err != nil {
- t.Fatalf("create client %s: %v", email, err)
- }
- }
- mkLink := func(clientID, inboundID int, flow string) {
- if err := db.Create(&model.ClientInbound{ClientId: clientID, InboundId: inboundID, FlowOverride: flow}).Error; err != nil {
- t.Fatalf("link %d/%d: %v", clientID, inboundID, err)
- }
- }
- mkClient(1, "vis@x")
- mkClient(2, "plain@x")
- mkLink(1, 20, "") // higher inbound_id, empty
- mkLink(1, 10, vision) // lower inbound_id, Vision
- mkLink(2, 30, "") // only empty override
- cs := &ClientService{}
- got, err := cs.EffectiveFlowsByEmails(nil, []string{"vis@x", "plain@x", "missing@x"})
- if err != nil {
- t.Fatalf("EffectiveFlowsByEmails: %v", err)
- }
- if got["vis@x"] != vision {
- t.Errorf("vis@x = %q, want %q (lowest inbound_id flow_override)", got["vis@x"], vision)
- }
- if v, ok := got["plain@x"]; ok {
- t.Errorf("plain@x present (%q); want absent (no non-empty flow anywhere)", v)
- }
- if v, ok := got["missing@x"]; ok {
- t.Errorf("missing@x present (%q); want absent (unknown client)", v)
- }
- // Empty input is a no-op (no query).
- if m, err := cs.EffectiveFlowsByEmails(nil, nil); err != nil || len(m) != 0 {
- t.Errorf("empty input: got %v err %v, want empty map", m, err)
- }
- }
|