init
This commit is contained in:
176
internal/collector/metrics.go
Normal file
176
internal/collector/metrics.go
Normal file
@@ -0,0 +1,176 @@
|
||||
package collector
|
||||
|
||||
import (
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"io"
|
||||
"net/http"
|
||||
"os"
|
||||
"strconv"
|
||||
"strings"
|
||||
"time"
|
||||
)
|
||||
|
||||
// XrayMetrics represents the metrics response from Xray's /debug/vars endpoint
|
||||
type XrayMetrics struct {
|
||||
Stats *XrayStats `json:"stats,omitempty"`
|
||||
}
|
||||
|
||||
// XrayStats contains inbound, outbound, and user traffic stats
|
||||
type XrayStats struct {
|
||||
Inbound map[string]TrafficData `json:"inbound,omitempty"`
|
||||
Outbound map[string]TrafficData `json:"outbound,omitempty"`
|
||||
User map[string]TrafficData `json:"user,omitempty"`
|
||||
}
|
||||
|
||||
// TrafficData contains uplink and downlink traffic in bytes
|
||||
type TrafficData struct {
|
||||
Uplink int64 `json:"uplink"`
|
||||
Downlink int64 `json:"downlink"`
|
||||
}
|
||||
|
||||
// XrayConfig represents the structure of xray config.json for reading metrics port
|
||||
type XrayConfig struct {
|
||||
Log json.RawMessage `json:"log,omitempty"`
|
||||
DNS json.RawMessage `json:"dns,omitempty"`
|
||||
API json.RawMessage `json:"api,omitempty"`
|
||||
Stats json.RawMessage `json:"stats,omitempty"`
|
||||
Policy json.RawMessage `json:"policy,omitempty"`
|
||||
Routing json.RawMessage `json:"routing,omitempty"`
|
||||
Inbounds json.RawMessage `json:"inbounds,omitempty"`
|
||||
Outbounds json.RawMessage `json:"outbounds,omitempty"`
|
||||
Metrics *MetricsConfig `json:"metrics,omitempty"`
|
||||
}
|
||||
|
||||
// MetricsConfig represents the metrics section in xray config
|
||||
type MetricsConfig struct {
|
||||
Tag string `json:"tag,omitempty"`
|
||||
Listen string `json:"listen,omitempty"` // Format: "127.0.0.1:38889"
|
||||
}
|
||||
|
||||
// Collector collects traffic metrics from Xray servers
|
||||
type Collector struct {
|
||||
httpClient *http.Client
|
||||
defaultMetricsPort int
|
||||
defaultMetricsHost string
|
||||
}
|
||||
|
||||
// NewCollector creates a new metrics collector
|
||||
func NewCollector() *Collector {
|
||||
return &Collector{
|
||||
httpClient: &http.Client{Timeout: 10 * time.Second},
|
||||
defaultMetricsPort: 38889,
|
||||
defaultMetricsHost: "127.0.0.1",
|
||||
}
|
||||
}
|
||||
|
||||
// GetMetricsPortFromConfig reads the metrics port from xray config file
|
||||
func (c *Collector) GetMetricsPortFromConfig(configPath string) (string, int, error) {
|
||||
if configPath == "" {
|
||||
return "127.0.0.1", c.defaultMetricsPort, nil
|
||||
}
|
||||
|
||||
data, err := os.ReadFile(configPath)
|
||||
if err != nil {
|
||||
return "127.0.0.1", c.defaultMetricsPort, fmt.Errorf("read config file: %w", err)
|
||||
}
|
||||
|
||||
var config XrayConfig
|
||||
if err := json.Unmarshal(data, &config); err != nil {
|
||||
return "127.0.0.1", c.defaultMetricsPort, fmt.Errorf("parse config file: %w", err)
|
||||
}
|
||||
|
||||
if config.Metrics == nil || config.Metrics.Listen == "" {
|
||||
return "", 0, fmt.Errorf("metrics not configured in xray config")
|
||||
}
|
||||
|
||||
// Parse listen address (format: "127.0.0.1:38889" or ":38889")
|
||||
listen := config.Metrics.Listen
|
||||
host := "127.0.0.1"
|
||||
var port int
|
||||
|
||||
if strings.Contains(listen, ":") {
|
||||
parts := strings.Split(listen, ":")
|
||||
if len(parts) == 2 {
|
||||
if parts[0] != "" {
|
||||
host = parts[0]
|
||||
}
|
||||
p, err := strconv.Atoi(parts[1])
|
||||
if err != nil {
|
||||
return "", 0, fmt.Errorf("invalid metrics port: %s", parts[1])
|
||||
}
|
||||
port = p
|
||||
}
|
||||
} else {
|
||||
// Try to parse as port only
|
||||
p, err := strconv.Atoi(listen)
|
||||
if err != nil {
|
||||
return "", 0, fmt.Errorf("invalid metrics listen format: %s", listen)
|
||||
}
|
||||
port = p
|
||||
}
|
||||
|
||||
if port <= 0 || port > 65535 {
|
||||
return "", 0, fmt.Errorf("invalid metrics port: %d", port)
|
||||
}
|
||||
|
||||
return host, port, nil
|
||||
}
|
||||
|
||||
// FetchMetrics fetches metrics from Xray's /debug/vars endpoint
|
||||
func (c *Collector) FetchMetrics(host string, port int) (*XrayMetrics, error) {
|
||||
url := fmt.Sprintf("http://%s:%d/debug/vars", host, port)
|
||||
|
||||
req, err := http.NewRequest("GET", url, nil)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("create request: %w", err)
|
||||
}
|
||||
|
||||
resp, err := c.httpClient.Do(req)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("http request: %w", err)
|
||||
}
|
||||
defer resp.Body.Close()
|
||||
|
||||
if resp.StatusCode != http.StatusOK {
|
||||
return nil, fmt.Errorf("unexpected status code: %d", resp.StatusCode)
|
||||
}
|
||||
|
||||
body, err := io.ReadAll(resp.Body)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("read body: %w", err)
|
||||
}
|
||||
|
||||
var metrics XrayMetrics
|
||||
if err := json.Unmarshal(body, &metrics); err != nil {
|
||||
return nil, fmt.Errorf("unmarshal metrics: %w", err)
|
||||
}
|
||||
|
||||
return &metrics, nil
|
||||
}
|
||||
|
||||
// MergeStats merges source stats into dest stats
|
||||
func MergeStats(dest, source *XrayStats) {
|
||||
if source == nil {
|
||||
return
|
||||
}
|
||||
if dest.Inbound == nil {
|
||||
dest.Inbound = make(map[string]TrafficData)
|
||||
}
|
||||
if dest.Outbound == nil {
|
||||
dest.Outbound = make(map[string]TrafficData)
|
||||
}
|
||||
if dest.User == nil {
|
||||
dest.User = make(map[string]TrafficData)
|
||||
}
|
||||
|
||||
for k, v := range source.Inbound {
|
||||
dest.Inbound[k] = v
|
||||
}
|
||||
for k, v := range source.Outbound {
|
||||
dest.Outbound[k] = v
|
||||
}
|
||||
for k, v := range source.User {
|
||||
dest.User[k] = v
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user