v2fly/app/stats/command/command.go

126 lines
2.8 KiB
Go
Raw Normal View History

2018-03-31 08:30:12 +00:00
package command
2021-02-16 20:31:50 +00:00
//go:generate go run github.com/v2fly/v2ray-core/v4/common/errors/errorgen
2018-03-31 08:30:12 +00:00
import (
"context"
2019-07-08 06:12:17 +00:00
"runtime"
"time"
2018-03-31 08:30:12 +00:00
grpc "google.golang.org/grpc"
2018-07-10 21:40:58 +00:00
2021-02-16 20:31:50 +00:00
core "github.com/v2fly/v2ray-core/v4"
"github.com/v2fly/v2ray-core/v4/app/stats"
"github.com/v2fly/v2ray-core/v4/common"
"github.com/v2fly/v2ray-core/v4/common/strmatcher"
feature_stats "github.com/v2fly/v2ray-core/v4/features/stats"
2018-03-31 08:30:12 +00:00
)
2018-07-10 21:40:58 +00:00
// statsServer is an implementation of StatsService.
2018-03-31 08:30:12 +00:00
type statsServer struct {
2019-07-08 06:12:17 +00:00
stats feature_stats.Manager
startTime time.Time
2018-03-31 08:30:12 +00:00
}
func NewStatsServer(manager feature_stats.Manager) StatsServiceServer {
2019-07-08 06:12:17 +00:00
return &statsServer{
stats: manager,
startTime: time.Now(),
}
2018-07-10 21:40:58 +00:00
}
2018-03-31 08:30:12 +00:00
func (s *statsServer) GetStats(ctx context.Context, request *GetStatsRequest) (*GetStatsResponse, error) {
c := s.stats.GetCounter(request.Name)
if c == nil {
return nil, newError(request.Name, " not found.")
}
var value int64
if request.Reset_ {
value = c.Set(0)
} else {
value = c.Value()
}
return &GetStatsResponse{
Stat: &Stat{
Name: request.Name,
Value: value,
},
}, nil
}
2018-07-10 21:40:58 +00:00
func (s *statsServer) QueryStats(ctx context.Context, request *QueryStatsRequest) (*QueryStatsResponse, error) {
matcher, err := strmatcher.Substr.New(request.Pattern)
if err != nil {
return nil, err
}
response := &QueryStatsResponse{}
manager, ok := s.stats.(*stats.Manager)
if !ok {
return nil, newError("QueryStats only works its own stats.Manager.")
}
manager.VisitCounters(func(name string, c feature_stats.Counter) bool {
2018-07-10 21:40:58 +00:00
if matcher.Match(name) {
var value int64
if request.Reset_ {
value = c.Set(0)
} else {
value = c.Value()
}
response.Stat = append(response.Stat, &Stat{
Name: name,
Value: value,
})
}
return true
})
return response, nil
}
2019-07-08 06:12:17 +00:00
func (s *statsServer) GetSysStats(ctx context.Context, request *SysStatsRequest) (*SysStatsResponse, error) {
var rtm runtime.MemStats
runtime.ReadMemStats(&rtm)
uptime := time.Since(s.startTime)
response := &SysStatsResponse{
Uptime: uint32(uptime.Seconds()),
NumGoroutine: uint32(runtime.NumGoroutine()),
Alloc: rtm.Alloc,
TotalAlloc: rtm.TotalAlloc,
Sys: rtm.Sys,
Mallocs: rtm.Mallocs,
Frees: rtm.Frees,
LiveObjects: rtm.Mallocs - rtm.Frees,
NumGC: rtm.NumGC,
PauseTotalNs: rtm.PauseTotalNs,
}
return response, nil
}
func (s *statsServer) mustEmbedUnimplementedStatsServiceServer() {}
2018-03-31 08:30:12 +00:00
type service struct {
2018-10-22 09:26:22 +00:00
statsManager feature_stats.Manager
2018-03-31 08:30:12 +00:00
}
func (s *service) Register(server *grpc.Server) {
2018-10-22 09:26:22 +00:00
RegisterStatsServiceServer(server, NewStatsServer(s.statsManager))
2018-03-31 08:30:12 +00:00
}
func init() {
common.Must(common.RegisterConfig((*Config)(nil), func(ctx context.Context, cfg interface{}) (interface{}, error) {
2018-10-22 09:26:22 +00:00
s := new(service)
core.RequireFeatures(ctx, func(sm feature_stats.Manager) {
s.statsManager = sm
})
return s, nil
2018-03-31 08:30:12 +00:00
}))
}