mirror of
https://github.com/apernet/hysteria.git
synced 2025-04-03 20:47:38 +03:00
add getOnline feature
This commit is contained in:
parent
c831b987cd
commit
415ef42b5a
1 changed files with 40 additions and 7 deletions
|
@ -5,6 +5,7 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
"strconv"
|
||||||
"sync"
|
"sync"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/apernet/hysteria/core/server"
|
"github.com/apernet/hysteria/core/server"
|
||||||
)
|
)
|
||||||
|
@ -22,17 +23,19 @@ type TrafficStatsServer interface {
|
||||||
|
|
||||||
func NewTrafficStatsServer(secret string) TrafficStatsServer {
|
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{}),
|
||||||
Secret: secret,
|
OnlineMap: make(map[string]int64),
|
||||||
|
Secret: secret,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type trafficStatsServerImpl struct {
|
type trafficStatsServerImpl struct {
|
||||||
Mutex sync.RWMutex
|
Mutex sync.RWMutex
|
||||||
StatsMap map[string]*trafficStatsEntry
|
StatsMap map[string]*trafficStatsEntry
|
||||||
KickMap map[string]struct{}
|
OnlineMap map[string]int64
|
||||||
Secret string
|
KickMap map[string]struct{}
|
||||||
|
Secret string
|
||||||
}
|
}
|
||||||
|
|
||||||
type trafficStatsEntry struct {
|
type trafficStatsEntry struct {
|
||||||
|
@ -58,6 +61,8 @@ 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
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,6 +83,10 @@ func (s *trafficStatsServerImpl) ServeHTTP(w http.ResponseWriter, r *http.Reques
|
||||||
s.kick(w, r)
|
s.kick(w, r)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
if r.Method == http.MethodPost && r.URL.Path == "/online" {
|
||||||
|
s.getOnline(w, r)
|
||||||
|
return
|
||||||
|
}
|
||||||
http.NotFound(w, r)
|
http.NotFound(w, r)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,6 +112,30 @@ func (s *trafficStatsServerImpl) getTraffic(w http.ResponseWriter, r *http.Reque
|
||||||
_, _ = w.Write(jb)
|
_, _ = w.Write(jb)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *trafficStatsServerImpl) getOnline(w http.ResponseWriter, r *http.Request) {
|
||||||
|
var jb []byte
|
||||||
|
var err error
|
||||||
|
|
||||||
|
timeNow := time.Now().Unix()
|
||||||
|
|
||||||
|
for id, lastSeen := range s.OnlineMap {
|
||||||
|
if timeNow-lastSeen > 180 {
|
||||||
|
delete(s.OnlineMap, id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
s.Mutex.RLock()
|
||||||
|
jb, err = json.Marshal(s.OnlineMap)
|
||||||
|
s.Mutex.RUnlock()
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
w.Header().Set("Content-Type", "application/json; charset=utf-8")
|
||||||
|
_, _ = w.Write(jb)
|
||||||
|
}
|
||||||
|
|
||||||
func (s *trafficStatsServerImpl) kick(w http.ResponseWriter, r *http.Request) {
|
func (s *trafficStatsServerImpl) kick(w http.ResponseWriter, r *http.Request) {
|
||||||
var ids []string
|
var ids []string
|
||||||
err := json.NewDecoder(r.Body).Decode(&ids)
|
err := json.NewDecoder(r.Body).Decode(&ids)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue