mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-03 20:47:38 +03:00
refactor getOnline feature
This commit is contained in:
parent
2366882bd6
commit
88eef7617f
3 changed files with 64 additions and 14 deletions
|
@ -196,4 +196,6 @@ type EventLogger interface {
|
||||||
// The implementation of this interface must be thread-safe.
|
// The implementation of this interface must be thread-safe.
|
||||||
type TrafficLogger interface {
|
type TrafficLogger interface {
|
||||||
Log(id string, tx, rx uint64) (ok bool)
|
Log(id string, tx, rx uint64) (ok bool)
|
||||||
|
LogOnline(id string, addr net.Addr)
|
||||||
|
LogOffline(id string, addr net.Addr)
|
||||||
}
|
}
|
||||||
|
|
|
@ -83,6 +83,7 @@ func (s *serverImpl) handleClient(conn quic.Connection) {
|
||||||
err := h3s.ServeQUICConn(conn)
|
err := h3s.ServeQUICConn(conn)
|
||||||
// If the client is authenticated, we need to log the disconnect event
|
// If the client is authenticated, we need to log the disconnect event
|
||||||
if handler.authenticated && s.config.EventLogger != nil {
|
if handler.authenticated && s.config.EventLogger != nil {
|
||||||
|
s.config.TrafficLogger.LogOffline(handler.authID, conn.RemoteAddr())
|
||||||
s.config.EventLogger.Disconnect(conn.RemoteAddr(), handler.authID, err)
|
s.config.EventLogger.Disconnect(conn.RemoteAddr(), handler.authID, err)
|
||||||
}
|
}
|
||||||
_ = conn.CloseWithError(closeErrCodeOK, "")
|
_ = conn.CloseWithError(closeErrCodeOK, "")
|
||||||
|
@ -154,6 +155,7 @@ func (h *h3sHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
w.WriteHeader(protocol.StatusAuthOK)
|
w.WriteHeader(protocol.StatusAuthOK)
|
||||||
// Call event logger
|
// Call event logger
|
||||||
if h.config.EventLogger != nil {
|
if h.config.EventLogger != nil {
|
||||||
|
h.config.TrafficLogger.LogOnline(id, h.conn.RemoteAddr())
|
||||||
h.config.EventLogger.Connect(h.conn.RemoteAddr(), id, actualTx)
|
h.config.EventLogger.Connect(h.conn.RemoteAddr(), id, actualTx)
|
||||||
}
|
}
|
||||||
// Initialize UDP session manager (if UDP is enabled)
|
// Initialize UDP session manager (if UDP is enabled)
|
||||||
|
|
|
@ -2,12 +2,11 @@ package trafficlogger
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"github.com/apernet/hysteria/core/server"
|
||||||
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/apernet/hysteria/core/server"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -25,7 +24,7 @@ func NewTrafficStatsServer(secret string) TrafficStatsServer {
|
||||||
return &trafficStatsServerImpl{
|
return &trafficStatsServerImpl{
|
||||||
StatsMap: make(map[string]*trafficStatsEntry),
|
StatsMap: make(map[string]*trafficStatsEntry),
|
||||||
KickMap: make(map[string]struct{}),
|
KickMap: make(map[string]struct{}),
|
||||||
OnlineMap: make(map[string]int64),
|
OnlineMap: make(map[string]map[string]bool),
|
||||||
Secret: secret,
|
Secret: secret,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -33,7 +32,7 @@ func NewTrafficStatsServer(secret string) TrafficStatsServer {
|
||||||
type trafficStatsServerImpl struct {
|
type trafficStatsServerImpl struct {
|
||||||
Mutex sync.RWMutex
|
Mutex sync.RWMutex
|
||||||
StatsMap map[string]*trafficStatsEntry
|
StatsMap map[string]*trafficStatsEntry
|
||||||
OnlineMap map[string]int64
|
OnlineMap map[string]map[string]bool
|
||||||
KickMap map[string]struct{}
|
KickMap map[string]struct{}
|
||||||
Secret string
|
Secret string
|
||||||
}
|
}
|
||||||
|
@ -61,11 +60,45 @@ func (s *trafficStatsServerImpl) Log(id string, tx, rx uint64) (ok bool) {
|
||||||
entry.Tx += tx
|
entry.Tx += tx
|
||||||
entry.Rx += rx
|
entry.Rx += rx
|
||||||
|
|
||||||
s.OnlineMap[id] = time.Now().Unix()
|
|
||||||
|
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// LogOnline adds the user to the online map.
|
||||||
|
func (s *trafficStatsServerImpl) LogOnline(id string, addr net.Addr) {
|
||||||
|
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 {
|
||||||
|
delete(s.OnlineMap, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
func (s *trafficStatsServerImpl) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
func (s *trafficStatsServerImpl) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||||
if s.Secret != "" && r.Header.Get("Authorization") != s.Secret {
|
if s.Secret != "" && r.Header.Get("Authorization") != s.Secret {
|
||||||
http.Error(w, "unauthorized", http.StatusUnauthorized)
|
http.Error(w, "unauthorized", http.StatusUnauthorized)
|
||||||
|
@ -116,17 +149,30 @@ func (s *trafficStatsServerImpl) getOnline(w http.ResponseWriter, r *http.Reques
|
||||||
var jb []byte
|
var jb []byte
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
timeNow := time.Now().Unix()
|
bClear, _ := strconv.ParseBool(r.URL.Query().Get("clear"))
|
||||||
|
OnlineSet := make(map[string][]string)
|
||||||
|
|
||||||
for id, lastSeen := range s.OnlineMap {
|
if bClear {
|
||||||
if timeNow-lastSeen > 180 {
|
s.Mutex.Lock()
|
||||||
delete(s.OnlineMap, id)
|
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()
|
||||||
}
|
}
|
||||||
|
|
||||||
s.Mutex.RLock()
|
jb, err = json.Marshal(OnlineSet)
|
||||||
jb, err = json.Marshal(s.OnlineMap)
|
|
||||||
s.Mutex.RUnlock()
|
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
http.Error(w, err.Error(), http.StatusInternalServerError)
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue