mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-07-01 10:04:19 +00:00
123 lines
3.4 KiB
Go
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)
|
|
}
|
|
}
|