| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- package service
- import (
- "testing"
- "github.com/mhsanaei/3x-ui/v3/internal/database"
- "github.com/mhsanaei/3x-ui/v3/internal/database/model"
- "github.com/mhsanaei/3x-ui/v3/internal/web/runtime"
- )
- // #4983: a synced inbound's OriginNodeGuid must point at the panel that
- // physically hosts it. A node's own local inbound (empty origin in its
- // snapshot) is attributed to the node's own GUID; an inbound the node forwards
- // from its own sub-node (non-empty origin) keeps that deeper GUID across the
- // hop — so a chained Node1->Node2->Node3 attributes Node3's inbounds to Node3.
- func TestSetRemoteTraffic_AttributesOriginNodeGuid(t *testing.T) {
- setupConflictDB(t)
- db := database.GetDB()
- const nodeID = 1
- if err := db.Create(&model.Node{
- Id: nodeID,
- Name: "node2",
- Address: "10.0.0.2",
- Port: 2053,
- ApiToken: "t",
- Guid: "node2-guid",
- }).Error; err != nil {
- t.Fatalf("create node: %v", err)
- }
- snap := &runtime.TrafficSnapshot{
- Inbounds: []*model.Inbound{
- { // node2's own local inbound — reports no origin
- Tag: "in-443-tcp",
- Enable: true,
- Port: 443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- },
- { // forwarded from node2's sub-node (node3) — carries node3's guid
- Tag: "in-8443-tcp",
- Enable: true,
- Port: 8443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- OriginNodeGuid: "node3-guid",
- },
- },
- }
- svc := InboundService{}
- if _, err := svc.setRemoteTrafficLocked(nodeID, snap, false); err != nil {
- t.Fatalf("setRemoteTrafficLocked: %v", err)
- }
- origin := func(tag string) string {
- var ib model.Inbound
- if err := db.Where("tag = ?", tag).First(&ib).Error; err != nil {
- t.Fatalf("load inbound %q: %v", tag, err)
- }
- return ib.OriginNodeGuid
- }
- if og := origin("in-443-tcp"); og != "node2-guid" {
- t.Fatalf("local inbound origin = %q, want node2-guid (the node's own GUID)", og)
- }
- if og := origin("in-8443-tcp"); og != "node3-guid" {
- t.Fatalf("forwarded inbound origin = %q, want node3-guid (kept across the hop)", og)
- }
- }
- // A cloned node reports its OWN inbound with its own (duplicated) panelGuid as
- // the origin. That must be remapped to the node-unique key, not stored verbatim
- // — otherwise origin_node_guid keeps the shared GUID while online is keyed by
- // the node-unique key, and the inbound page reads an empty bucket (shows
- // offline). A genuinely forwarded sub-node GUID is still kept across the hop.
- func TestSetRemoteTraffic_RemapsClonedNodeOwnGuidOrigin(t *testing.T) {
- setupConflictDB(t)
- db := database.GetDB()
- // Two nodes share one panelGuid (cloned servers).
- for _, n := range []*model.Node{
- {Id: 1, Name: "a", Address: "10.0.0.1", Port: 2053, ApiToken: "t", Guid: "dup"},
- {Id: 2, Name: "b", Address: "10.0.0.2", Port: 2053, ApiToken: "t", Guid: "dup"},
- } {
- if err := db.Create(n).Error; err != nil {
- t.Fatalf("create node %s: %v", n.Name, err)
- }
- }
- snap := &runtime.TrafficSnapshot{
- Inbounds: []*model.Inbound{
- { // node 1's OWN inbound, reporting its own (shared) panelGuid as origin
- Tag: "own-443-tcp",
- Enable: true,
- Port: 443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- OriginNodeGuid: "dup",
- },
- { // forwarded from a sub-node with a distinct guid — kept across the hop
- Tag: "fwd-8443-tcp",
- Enable: true,
- Port: 8443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- OriginNodeGuid: "child-guid",
- },
- },
- }
- svc := InboundService{}
- if _, err := svc.setRemoteTrafficLocked(1, snap, false); err != nil {
- t.Fatalf("setRemoteTrafficLocked: %v", err)
- }
- origin := func(tag string) string {
- var ib model.Inbound
- if err := db.Where("tag = ?", tag).First(&ib).Error; err != nil {
- t.Fatalf("load inbound %q: %v", tag, err)
- }
- return ib.OriginNodeGuid
- }
- if og := origin("own-443-tcp"); og != "node:1" {
- t.Fatalf("cloned node's own inbound origin = %q, want node:1 (remapped from shared GUID)", og)
- }
- if og := origin("fwd-8443-tcp"); og != "child-guid" {
- t.Fatalf("forwarded inbound origin = %q, want child-guid (kept across the hop)", og)
- }
- }
- func TestSetRemoteTraffic_PreservesLocalShareAddressStrategy(t *testing.T) {
- setupConflictDB(t)
- db := database.GetDB()
- const nodeID = 1
- if err := db.Create(&model.Node{
- Id: nodeID,
- Name: "node2",
- Address: "10.0.0.2",
- Port: 2053,
- ApiToken: "t",
- Guid: "node2-guid",
- }).Error; err != nil {
- t.Fatalf("create node: %v", err)
- }
- nodeIDPtr := nodeID
- if err := db.Create(&model.Inbound{
- UserId: 1,
- NodeID: &nodeIDPtr,
- Tag: "remote-in",
- Enable: true,
- Port: 443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- ShareAddrStrategy: "custom",
- ShareAddr: "edge.example.com",
- }).Error; err != nil {
- t.Fatalf("create central inbound: %v", err)
- }
- snap := &runtime.TrafficSnapshot{
- Inbounds: []*model.Inbound{{
- Tag: "remote-in",
- Enable: true,
- Port: 8443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- }},
- }
- svc := InboundService{}
- if _, err := svc.setRemoteTrafficLocked(nodeID, snap, false); err != nil {
- t.Fatalf("setRemoteTrafficLocked: %v", err)
- }
- var ib model.Inbound
- if err := db.Where("tag = ?", "remote-in").First(&ib).Error; err != nil {
- t.Fatalf("load inbound: %v", err)
- }
- if ib.ShareAddrStrategy != "custom" || ib.ShareAddr != "edge.example.com" {
- t.Fatalf("share address fields were overwritten: strategy=%q addr=%q", ib.ShareAddrStrategy, ib.ShareAddr)
- }
- if ib.Port != 8443 {
- t.Fatalf("sync should still update regular remote fields; port = %d, want 8443", ib.Port)
- }
- }
- func TestSetRemoteTraffic_DefaultsShareAddressFieldsForNewCentralInbound(t *testing.T) {
- setupConflictDB(t)
- db := database.GetDB()
- const nodeID = 1
- if err := db.Create(&model.Node{
- Id: nodeID,
- Name: "node2",
- Address: "10.0.0.2",
- Port: 2053,
- ApiToken: "t",
- Guid: "node2-guid",
- }).Error; err != nil {
- t.Fatalf("create node: %v", err)
- }
- snap := &runtime.TrafficSnapshot{
- Inbounds: []*model.Inbound{{
- Tag: "remote-in",
- Enable: true,
- Port: 8443,
- Protocol: model.VLESS,
- Settings: `{"clients":[]}`,
- ShareAddrStrategy: "custom",
- ShareAddr: "remote.example.com",
- }},
- }
- svc := InboundService{}
- if _, err := svc.setRemoteTrafficLocked(nodeID, snap, false); err != nil {
- t.Fatalf("setRemoteTrafficLocked: %v", err)
- }
- var ib model.Inbound
- if err := db.Where("tag = ?", "remote-in").First(&ib).Error; err != nil {
- t.Fatalf("load inbound: %v", err)
- }
- if ib.ShareAddrStrategy != "node" || ib.ShareAddr != "" {
- t.Fatalf("new central inbound share fields = (%q, %q), want (node, empty)", ib.ShareAddrStrategy, ib.ShareAddr)
- }
- }
|