mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-05 11:59:35 +00:00
fix(online): scope online status per node instead of a global union
The inbounds page and Nodes page checked each client's email against a single deduped union of every node's online clients, so a client connected to one node showed as online on every inbound across every node. The local online set was also derived from the email-keyed client_traffics.last_online column, which remote-node syncs bump too, leaking remote-only clients onto local inbounds. Track online clients per node: the local panel's own xray clients under key 0 (derived from live traffic-poll deltas via RefreshLocalOnline, kept in memory and independent of the shared last_online column) and each remote node under its id. Add GetOnlineClientsByNode plus a /clients/onlinesByNode endpoint and onlineByNode WS field; node.go and the inbounds rollup now scope online by node. The flat GetOnlineClients union is kept for client-centric and total-count views (Clients page, dashboard, telegram). Closes #4809
This commit is contained in:
@@ -224,10 +224,7 @@ func (s *NodeService) GetAll() ([]*model.Node, error) {
|
||||
Select("inbound_id, email, enable, total, up, down, expiry_time").
|
||||
Where("inbound_id IN ?", inboundIDs).
|
||||
Scan(&trafficRows).Error; err == nil {
|
||||
online := make(map[string]struct{})
|
||||
for _, email := range s.onlineEmails() {
|
||||
online[email] = struct{}{}
|
||||
}
|
||||
onlineByNodeSet := s.onlineEmailsByNode()
|
||||
depletedByNode := make(map[int]int)
|
||||
onlineByNode := make(map[int]int)
|
||||
for _, row := range trafficRows {
|
||||
@@ -240,8 +237,12 @@ func (s *NodeService) GetAll() ([]*model.Node, error) {
|
||||
if expired || exhausted || !row.Enable {
|
||||
depletedByNode[nodeID]++
|
||||
}
|
||||
if _, ok := online[row.Email]; ok {
|
||||
onlineByNode[nodeID]++
|
||||
// Scope online by the node the inbound lives on: a client online
|
||||
// on one node must not count as online on another.
|
||||
if set, ok := onlineByNodeSet[nodeID]; ok {
|
||||
if _, isOnline := set[row.Email]; isOnline {
|
||||
onlineByNode[nodeID]++
|
||||
}
|
||||
}
|
||||
}
|
||||
for _, n := range nodes {
|
||||
@@ -254,9 +255,18 @@ func (s *NodeService) GetAll() ([]*model.Node, error) {
|
||||
return nodes, nil
|
||||
}
|
||||
|
||||
func (s *NodeService) onlineEmails() []string {
|
||||
func (s *NodeService) onlineEmailsByNode() map[int]map[string]struct{} {
|
||||
svc := InboundService{}
|
||||
return svc.GetOnlineClients()
|
||||
byNode := svc.GetOnlineClientsByNode()
|
||||
out := make(map[int]map[string]struct{}, len(byNode))
|
||||
for nodeID, emails := range byNode {
|
||||
set := make(map[string]struct{}, len(emails))
|
||||
for _, email := range emails {
|
||||
set[email] = struct{}{}
|
||||
}
|
||||
out[nodeID] = set
|
||||
}
|
||||
return out
|
||||
}
|
||||
|
||||
func (s *NodeService) GetById(id int) (*model.Node, error) {
|
||||
|
||||
Reference in New Issue
Block a user