Files
3x-ui/internal/web/runtime/manager_convergence_test.go
T

123 lines
3.4 KiB
Go

package runtime
import (
"testing"
"github.com/mhsanaei/3x-ui/v3/internal/database/model"
)
func TestManagerRemoteForRefreshesChangedCredential(t *testing.T) {
m := NewManager(LocalDeps{})
first, err := m.RemoteFor(&model.Node{
Id: 1,
Name: "node",
Scheme: "https",
Address: "node.example.com",
Port: 2053,
BasePath: "/",
ApiToken: "old-token",
})
if err != nil {
t.Fatalf("first RemoteFor: %v", err)
}
second, err := m.RemoteFor(&model.Node{
Id: 1,
Name: "node",
Scheme: "https",
Address: "node.example.com",
Port: 2053,
BasePath: "/",
ApiToken: "new-token",
})
if err != nil {
t.Fatalf("second RemoteFor: %v", err)
}
if second == first {
t.Fatal("RemoteFor reused stale Remote after ApiToken changed")
}
if got := second.node.ApiToken; got != "new-token" {
t.Fatalf("cached Remote token = %q, want new-token", got)
}
}
func TestManagerRemoteForIdentityFields(t *testing.T) {
base := model.Node{
Id: 7,
Name: "node-a",
Remark: "old remark",
Scheme: "https",
Address: "node.example.com",
Port: 2053,
BasePath: "/",
ApiToken: "token",
AllowPrivateAddress: true,
TlsVerifyMode: "pin",
PinnedCertSha256: "sha",
OutboundTag: "warp",
Status: "online",
InboundCount: 1,
}
cases := []struct {
name string
mutate func(*model.Node)
refresh bool
}{
{"same", func(*model.Node) {}, false},
{"name does not churn", func(n *model.Node) { n.Name = "renamed" }, false},
{"remark does not churn", func(n *model.Node) { n.Remark = "new remark" }, false},
{"status does not churn", func(n *model.Node) { n.Status = "offline" }, false},
{"metrics do not churn", func(n *model.Node) { n.InboundCount = 99 }, false},
{"scheme", func(n *model.Node) { n.Scheme = "http" }, true},
{"address", func(n *model.Node) { n.Address = "other.example.com" }, true},
{"port", func(n *model.Node) { n.Port = 8443 }, true},
{"base path", func(n *model.Node) { n.BasePath = "/x/" }, true},
{"api token", func(n *model.Node) { n.ApiToken = "next" }, true},
{"allow private", func(n *model.Node) { n.AllowPrivateAddress = false }, true},
{"tls verify mode", func(n *model.Node) { n.TlsVerifyMode = "skip" }, true},
{"pin", func(n *model.Node) { n.PinnedCertSha256 = "other" }, true},
{"outbound tag", func(n *model.Node) { n.OutboundTag = "direct" }, true},
}
for _, tc := range cases {
t.Run(tc.name, func(t *testing.T) {
m := NewManager(LocalDeps{})
firstNode := base
first, err := m.RemoteFor(&firstNode)
if err != nil {
t.Fatalf("first RemoteFor: %v", err)
}
nextNode := base
tc.mutate(&nextNode)
second, err := m.RemoteFor(&nextNode)
if err != nil {
t.Fatalf("second RemoteFor: %v", err)
}
if gotRefresh := second != first; gotRefresh != tc.refresh {
t.Fatalf("refresh = %v, want %v", gotRefresh, tc.refresh)
}
})
}
}
func TestManagerRemoteForClonesInputNode(t *testing.T) {
m := NewManager(LocalDeps{})
n := &model.Node{
Id: 9,
Scheme: "https",
Address: "node.example.com",
Port: 2053,
BasePath: "/",
ApiToken: "original",
}
rt, err := m.RemoteFor(n)
if err != nil {
t.Fatalf("RemoteFor: %v", err)
}
n.ApiToken = "mutated-after-cache"
if got := rt.node.ApiToken; got != "original" {
t.Fatalf("cached Remote observed caller mutation: %q", got)
}
}