mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2025-12-19 08:20:05 +08:00
Some checks are pending
Test / test (1.20, macos-15-intel) (push) Waiting to run
Test / test (1.20, macos-latest) (push) Waiting to run
Test / test (1.20, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.20, ubuntu-latest) (push) Waiting to run
Test / test (1.20, windows-latest) (push) Waiting to run
Test / test (1.21, macos-15-intel) (push) Waiting to run
Test / test (1.21, macos-latest) (push) Waiting to run
Test / test (1.21, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.21, ubuntu-latest) (push) Waiting to run
Test / test (1.21, windows-latest) (push) Waiting to run
Test / test (1.22, macos-15-intel) (push) Waiting to run
Test / test (1.22, macos-latest) (push) Waiting to run
Test / test (1.22, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.22, ubuntu-latest) (push) Waiting to run
Test / test (1.22, windows-latest) (push) Waiting to run
Test / test (1.23, macos-15-intel) (push) Waiting to run
Test / test (1.23, macos-latest) (push) Waiting to run
Test / test (1.23, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.23, ubuntu-latest) (push) Waiting to run
Test / test (1.23, windows-latest) (push) Waiting to run
Test / test (1.24, macos-15-intel) (push) Waiting to run
Test / test (1.24, macos-latest) (push) Waiting to run
Test / test (1.24, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.24, ubuntu-latest) (push) Waiting to run
Test / test (1.24, windows-latest) (push) Waiting to run
Test / test (1.25, macos-15-intel) (push) Waiting to run
Test / test (1.25, macos-latest) (push) Waiting to run
Test / test (1.25, ubuntu-24.04-arm) (push) Waiting to run
Test / test (1.25, ubuntu-latest) (push) Waiting to run
Test / test (1.25, windows-latest) (push) Waiting to run
Trigger CMFA Update / trigger-CMFA-update (push) Waiting to run
247 lines
5.5 KiB
Go
247 lines
5.5 KiB
Go
package statistic
|
|
|
|
import (
|
|
"io"
|
|
"net"
|
|
"time"
|
|
|
|
"github.com/metacubex/mihomo/common/atomic"
|
|
"github.com/metacubex/mihomo/common/buf"
|
|
N "github.com/metacubex/mihomo/common/net"
|
|
"github.com/metacubex/mihomo/common/utils"
|
|
C "github.com/metacubex/mihomo/constant"
|
|
|
|
"github.com/gofrs/uuid/v5"
|
|
)
|
|
|
|
type Tracker interface {
|
|
ID() string
|
|
Close() error
|
|
Info() *TrackerInfo
|
|
C.Connection
|
|
}
|
|
|
|
type TrackerInfo struct {
|
|
UUID uuid.UUID `json:"id"`
|
|
Metadata *C.Metadata `json:"metadata"`
|
|
UploadTotal atomic.Int64 `json:"upload"`
|
|
DownloadTotal atomic.Int64 `json:"download"`
|
|
Start time.Time `json:"start"`
|
|
Chain C.Chain `json:"chains"`
|
|
ProviderChain C.Chain `json:"providerChains"`
|
|
Rule string `json:"rule"`
|
|
RulePayload string `json:"rulePayload"`
|
|
}
|
|
|
|
type tcpTracker struct {
|
|
C.Conn `json:"-"`
|
|
*TrackerInfo
|
|
manager *Manager
|
|
|
|
pushToManager bool `json:"-"`
|
|
}
|
|
|
|
func (tt *tcpTracker) ID() string {
|
|
return tt.UUID.String()
|
|
}
|
|
|
|
func (tt *tcpTracker) Info() *TrackerInfo {
|
|
return tt.TrackerInfo
|
|
}
|
|
|
|
func (tt *tcpTracker) Read(b []byte) (int, error) {
|
|
n, err := tt.Conn.Read(b)
|
|
download := int64(n)
|
|
if tt.pushToManager {
|
|
tt.manager.PushDownloaded(download)
|
|
}
|
|
tt.DownloadTotal.Add(download)
|
|
return n, err
|
|
}
|
|
|
|
func (tt *tcpTracker) ReadBuffer(buffer *buf.Buffer) (err error) {
|
|
err = tt.Conn.ReadBuffer(buffer)
|
|
download := int64(buffer.Len())
|
|
if tt.pushToManager {
|
|
tt.manager.PushDownloaded(download)
|
|
}
|
|
tt.DownloadTotal.Add(download)
|
|
return
|
|
}
|
|
|
|
func (tt *tcpTracker) UnwrapReader() (io.Reader, []N.CountFunc) {
|
|
return tt.Conn, []N.CountFunc{func(download int64) {
|
|
if tt.pushToManager {
|
|
tt.manager.PushDownloaded(download)
|
|
}
|
|
tt.DownloadTotal.Add(download)
|
|
}}
|
|
}
|
|
|
|
func (tt *tcpTracker) Write(b []byte) (int, error) {
|
|
n, err := tt.Conn.Write(b)
|
|
upload := int64(n)
|
|
if tt.pushToManager {
|
|
tt.manager.PushUploaded(upload)
|
|
}
|
|
tt.UploadTotal.Add(upload)
|
|
return n, err
|
|
}
|
|
|
|
func (tt *tcpTracker) WriteBuffer(buffer *buf.Buffer) (err error) {
|
|
upload := int64(buffer.Len())
|
|
err = tt.Conn.WriteBuffer(buffer)
|
|
if tt.pushToManager {
|
|
tt.manager.PushUploaded(upload)
|
|
}
|
|
tt.UploadTotal.Add(upload)
|
|
return
|
|
}
|
|
|
|
func (tt *tcpTracker) UnwrapWriter() (io.Writer, []N.CountFunc) {
|
|
return tt.Conn, []N.CountFunc{func(upload int64) {
|
|
if tt.pushToManager {
|
|
tt.manager.PushUploaded(upload)
|
|
}
|
|
tt.UploadTotal.Add(upload)
|
|
}}
|
|
}
|
|
|
|
func (tt *tcpTracker) Close() error {
|
|
tt.manager.Leave(tt)
|
|
return tt.Conn.Close()
|
|
}
|
|
|
|
func (tt *tcpTracker) Upstream() any {
|
|
return tt.Conn
|
|
}
|
|
|
|
func NewTCPTracker(conn C.Conn, manager *Manager, metadata *C.Metadata, rule C.Rule, uploadTotal int64, downloadTotal int64, pushToManager bool) *tcpTracker {
|
|
metadata.RemoteDst = conn.RemoteDestination()
|
|
|
|
t := &tcpTracker{
|
|
Conn: conn,
|
|
manager: manager,
|
|
TrackerInfo: &TrackerInfo{
|
|
UUID: utils.NewUUIDV4(),
|
|
Start: time.Now(),
|
|
Metadata: metadata,
|
|
Chain: conn.Chains(),
|
|
ProviderChain: conn.ProviderChains(),
|
|
Rule: "",
|
|
UploadTotal: atomic.NewInt64(uploadTotal),
|
|
DownloadTotal: atomic.NewInt64(downloadTotal),
|
|
},
|
|
pushToManager: pushToManager,
|
|
}
|
|
|
|
if pushToManager {
|
|
if uploadTotal > 0 {
|
|
manager.PushUploaded(uploadTotal)
|
|
}
|
|
if downloadTotal > 0 {
|
|
manager.PushDownloaded(downloadTotal)
|
|
}
|
|
}
|
|
|
|
if rule != nil {
|
|
t.TrackerInfo.Rule = rule.RuleType().String()
|
|
t.TrackerInfo.RulePayload = rule.Payload()
|
|
}
|
|
|
|
manager.Join(t)
|
|
return t
|
|
}
|
|
|
|
type udpTracker struct {
|
|
C.PacketConn `json:"-"`
|
|
*TrackerInfo
|
|
manager *Manager
|
|
|
|
pushToManager bool `json:"-"`
|
|
}
|
|
|
|
func (ut *udpTracker) ID() string {
|
|
return ut.UUID.String()
|
|
}
|
|
|
|
func (ut *udpTracker) Info() *TrackerInfo {
|
|
return ut.TrackerInfo
|
|
}
|
|
|
|
func (ut *udpTracker) ReadFrom(b []byte) (int, net.Addr, error) {
|
|
n, addr, err := ut.PacketConn.ReadFrom(b)
|
|
download := int64(n)
|
|
if ut.pushToManager {
|
|
ut.manager.PushDownloaded(download)
|
|
}
|
|
ut.DownloadTotal.Add(download)
|
|
return n, addr, err
|
|
}
|
|
|
|
func (ut *udpTracker) WaitReadFrom() (data []byte, put func(), addr net.Addr, err error) {
|
|
data, put, addr, err = ut.PacketConn.WaitReadFrom()
|
|
download := int64(len(data))
|
|
if ut.pushToManager {
|
|
ut.manager.PushDownloaded(download)
|
|
}
|
|
ut.DownloadTotal.Add(download)
|
|
return
|
|
}
|
|
|
|
func (ut *udpTracker) WriteTo(b []byte, addr net.Addr) (int, error) {
|
|
n, err := ut.PacketConn.WriteTo(b, addr)
|
|
upload := int64(n)
|
|
if ut.pushToManager {
|
|
ut.manager.PushUploaded(upload)
|
|
}
|
|
ut.UploadTotal.Add(upload)
|
|
return n, err
|
|
}
|
|
|
|
func (ut *udpTracker) Close() error {
|
|
ut.manager.Leave(ut)
|
|
return ut.PacketConn.Close()
|
|
}
|
|
|
|
func (ut *udpTracker) Upstream() any {
|
|
return ut.PacketConn
|
|
}
|
|
|
|
func NewUDPTracker(conn C.PacketConn, manager *Manager, metadata *C.Metadata, rule C.Rule, uploadTotal int64, downloadTotal int64, pushToManager bool) *udpTracker {
|
|
metadata.RemoteDst = conn.RemoteDestination()
|
|
|
|
ut := &udpTracker{
|
|
PacketConn: conn,
|
|
manager: manager,
|
|
TrackerInfo: &TrackerInfo{
|
|
UUID: utils.NewUUIDV4(),
|
|
Start: time.Now(),
|
|
Metadata: metadata,
|
|
Chain: conn.Chains(),
|
|
ProviderChain: conn.ProviderChains(),
|
|
Rule: "",
|
|
UploadTotal: atomic.NewInt64(uploadTotal),
|
|
DownloadTotal: atomic.NewInt64(downloadTotal),
|
|
},
|
|
pushToManager: pushToManager,
|
|
}
|
|
|
|
if pushToManager {
|
|
if uploadTotal > 0 {
|
|
manager.PushUploaded(uploadTotal)
|
|
}
|
|
if downloadTotal > 0 {
|
|
manager.PushDownloaded(downloadTotal)
|
|
}
|
|
}
|
|
|
|
if rule != nil {
|
|
ut.TrackerInfo.Rule = rule.RuleType().String()
|
|
ut.TrackerInfo.RulePayload = rule.Payload()
|
|
}
|
|
|
|
manager.Join(ut)
|
|
return ut
|
|
}
|