package statistic import ( "io" "net" "time" "github.com/Dreamacro/clash/common/atomic" "github.com/Dreamacro/clash/common/buf" N "github.com/Dreamacro/clash/common/net" "github.com/Dreamacro/clash/common/utils" C "github.com/Dreamacro/clash/constant" "github.com/gofrs/uuid/v5" ) type tracker interface { ID() string Close() error 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"` 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) 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 { if conn != nil { if tcpAddr, ok := conn.RemoteAddr().(*net.TCPAddr); ok && tcpAddr.IP != nil { metadata.RemoteDst = tcpAddr.IP.String() } else { metadata.RemoteDst = conn.RemoteDestination() } } t := &tcpTracker{ Conn: conn, manager: manager, trackerInfo: &trackerInfo{ UUID: utils.NewUUIDV4(), Start: time.Now(), Metadata: metadata, Chain: conn.Chains(), 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) 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) 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 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(), 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 }