diff --git a/core/server/config.go b/core/server/config.go index fe64201..c630abc 100644 --- a/core/server/config.go +++ b/core/server/config.go @@ -196,6 +196,5 @@ type EventLogger interface { // The implementation of this interface must be thread-safe. type TrafficLogger interface { Log(id string, tx, rx uint64) (ok bool) - LogOnline(id string, addr net.Addr) - LogOffline(id string, addr net.Addr) + LogOnlineStateChanged(id string, online bool) } diff --git a/core/server/server.go b/core/server/server.go index 99351b0..5daf36e 100644 --- a/core/server/server.go +++ b/core/server/server.go @@ -83,7 +83,7 @@ func (s *serverImpl) handleClient(conn quic.Connection) { err := h3s.ServeQUICConn(conn) // If the client is authenticated, we need to log the disconnect event if handler.authenticated && s.config.EventLogger != nil { - s.config.TrafficLogger.LogOffline(handler.authID, conn.RemoteAddr()) + s.config.TrafficLogger.LogOnlineStateChanged(handler.authID, false) s.config.EventLogger.Disconnect(conn.RemoteAddr(), handler.authID, err) } _ = conn.CloseWithError(closeErrCodeOK, "") @@ -155,7 +155,7 @@ func (h *h3sHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { w.WriteHeader(protocol.StatusAuthOK) // Call event logger if h.config.EventLogger != nil { - h.config.TrafficLogger.LogOnline(id, h.conn.RemoteAddr()) + h.config.TrafficLogger.LogOnlineStateChanged(id, true) h.config.EventLogger.Connect(h.conn.RemoteAddr(), id, actualTx) } // Initialize UDP session manager (if UDP is enabled) diff --git a/extras/trafficlogger/http.go b/extras/trafficlogger/http.go index 5f46e0e..827d64d 100644 --- a/extras/trafficlogger/http.go +++ b/extras/trafficlogger/http.go @@ -2,11 +2,11 @@ package trafficlogger import ( "encoding/json" - "github.com/apernet/hysteria/core/server" - "net" "net/http" "strconv" "sync" + + "github.com/apernet/hysteria/core/server" ) const ( @@ -24,7 +24,7 @@ func NewTrafficStatsServer(secret string) TrafficStatsServer { return &trafficStatsServerImpl{ StatsMap: make(map[string]*trafficStatsEntry), KickMap: make(map[string]struct{}), - OnlineMap: make(map[string]map[string]bool), + OnlineMap: make(map[string]int), Secret: secret, } } @@ -32,7 +32,7 @@ func NewTrafficStatsServer(secret string) TrafficStatsServer { type trafficStatsServerImpl struct { Mutex sync.RWMutex StatsMap map[string]*trafficStatsEntry - OnlineMap map[string]map[string]bool + OnlineMap map[string]int KickMap map[string]struct{} Secret string } @@ -63,40 +63,19 @@ func (s *trafficStatsServerImpl) Log(id string, tx, rx uint64) (ok bool) { return true } -// LogOnline adds the user to the online map. -func (s *trafficStatsServerImpl) LogOnline(id string, addr net.Addr) { +// LogOnlineStateChanged updates the online state to the online map. +func (s *trafficStatsServerImpl) LogOnlineStateChanged(id string, online bool) { s.Mutex.Lock() defer s.Mutex.Unlock() - if _, ok := s.OnlineMap[id]; !ok { - s.OnlineMap[id] = make(map[string]bool) - } - userIp, _, err := net.SplitHostPort(addr.String()) - if err != nil { - return - } - s.OnlineMap[id][userIp] = true -} -// LogOffline removes the user from the online map. -func (s *trafficStatsServerImpl) LogOffline(id string, addr net.Addr) { - s.Mutex.Lock() - defer s.Mutex.Unlock() - userIp, _, err := net.SplitHostPort(addr.String()) - if err != nil { - return - } - if onlineUsers, ok := s.OnlineMap[id]; ok { - if !onlineUsers[userIp] { - //if the user's ip is not in the online map, delete the whole entry - delete(s.OnlineMap, id) - return - } - delete(onlineUsers, userIp) - if len(onlineUsers) == 0 { + if online { + s.OnlineMap[id]++ + } else { + s.OnlineMap[id]-- + if s.OnlineMap[id] <= 0 { delete(s.OnlineMap, id) } } - } func (s *trafficStatsServerImpl) ServeHTTP(w http.ResponseWriter, r *http.Request) { @@ -146,34 +125,10 @@ func (s *trafficStatsServerImpl) getTraffic(w http.ResponseWriter, r *http.Reque } func (s *trafficStatsServerImpl) getOnline(w http.ResponseWriter, r *http.Request) { - var jb []byte - var err error - - bClear, _ := strconv.ParseBool(r.URL.Query().Get("clear")) - OnlineSet := make(map[string][]string) - - if bClear { - s.Mutex.Lock() - for id, addrs := range s.OnlineMap { - for addr := range addrs { - OnlineSet[id] = append(OnlineSet[id], addr) - } - } - s.OnlineMap = make(map[string]map[string]bool) - s.Mutex.Unlock() - - } else { - s.Mutex.RLock() - for id, addrs := range s.OnlineMap { - for addr := range addrs { - OnlineSet[id] = append(OnlineSet[id], addr) - } - } - s.Mutex.RUnlock() - } - - jb, err = json.Marshal(OnlineSet) + s.Mutex.RLock() + defer s.Mutex.RUnlock() + jb, err := json.Marshal(s.OnlineMap) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) return