package stats import ( "errors" "log" "time" "github.com/anacrolix/torrent" ) var ErrTorrentNotFound = errors.New("torrent not found") type PieceStatus string const ( Checking PieceStatus = "H" Partial PieceStatus = "P" Complete PieceStatus = "C" Waiting PieceStatus = "W" Error PieceStatus = "?" ) type PieceChunk struct { Status PieceStatus `json:"status"` NumPieces int `json:"numPieces"` } type TorrentStats struct { Name string `json:"name"` Hash string `json:"hash"` DownloadedBytes int64 `json:"downloadedBytes"` UploadedBytes int64 `json:"uploadedBytes"` TimePassed float64 `json:"timePassed"` PieceChunks []*PieceChunk `json:"pieceChunks"` TotalPieces int `json:"totalPieces"` } type GlobalTorrentStats struct { DownloadedBytes int64 `json:"downloadedBytes"` UploadedBytes int64 `json:"uploadedBytes"` TimePassed float64 `json:"timePassed"` } type RouteStats struct { Name string `json:"name"` TorrentStats []*TorrentStats `json:"torrentStats"` } type stats struct { totalDownloadBytes int64 downloadBytes int64 totalUploadBytes int64 uploadBytes int64 time time.Time } type Torrent struct { torrents map[string]*torrent.Torrent torrentsByRoute map[string][]*torrent.Torrent previousStats map[string]*stats gTime time.Time } func NewTorrent() *Torrent { return &Torrent{ gTime: time.Now(), torrents: make(map[string]*torrent.Torrent), torrentsByRoute: make(map[string][]*torrent.Torrent), previousStats: make(map[string]*stats), } } func (s *Torrent) Add(route string, t *torrent.Torrent) { s.torrents[t.InfoHash().String()] = t s.previousStats[t.InfoHash().String()] = &stats{} tbr := s.torrentsByRoute[route] s.torrentsByRoute[route] = append(tbr, t) } func (s *Torrent) Stats(hash string) (*TorrentStats, error) { t, ok := s.torrents[hash] if !(ok) { return nil, ErrTorrentNotFound } now := time.Now() return s.stats(now, t, true), nil } func (s *Torrent) RoutesStats() []*RouteStats { now := time.Now() var out []*RouteStats for r, tl := range s.torrentsByRoute { var tStats []*TorrentStats for _, t := range tl { ts := s.stats(now, t, true) tStats = append(tStats, ts) } rs := &RouteStats{ Name: r, TorrentStats: tStats, } out = append(out, rs) } return out } func (s *Torrent) GlobalStats() *GlobalTorrentStats { now := time.Now() var totalDownload int64 var totalUpload int64 for _, torrent := range s.torrents { tStats := s.stats(now, torrent, false) totalDownload += tStats.DownloadedBytes totalUpload += tStats.UploadedBytes } timePassed := now.Sub(s.gTime) s.gTime = now return &GlobalTorrentStats{ DownloadedBytes: totalDownload, UploadedBytes: totalUpload, TimePassed: timePassed.Seconds(), } } func (s *Torrent) stats(now time.Time, t *torrent.Torrent, chunks bool) *TorrentStats { ts := &TorrentStats{} prev := s.previousStats[t.InfoHash().String()] if s.returnPreviousMeasurements(now) { ts.DownloadedBytes = prev.downloadBytes ts.UploadedBytes = prev.uploadBytes log.Println("Using previous stats") } else { st := t.Stats() rd := st.BytesReadData.Int64() wd := st.BytesWrittenData.Int64() ist := &stats{ downloadBytes: rd - prev.totalDownloadBytes, uploadBytes: wd - prev.totalUploadBytes, totalDownloadBytes: rd, totalUploadBytes: wd, time: now, } ts.DownloadedBytes = ist.downloadBytes ts.UploadedBytes = ist.uploadBytes s.previousStats[t.InfoHash().String()] = ist } ts.TimePassed = now.Sub(prev.time).Seconds() var totalPieces int if chunks { var pch []*PieceChunk for _, psr := range t.PieceStateRuns() { var s PieceStatus switch { case psr.Checking: s = Checking case psr.Partial: s = Partial case psr.Complete: s = Complete case !psr.Ok: s = Error default: s = Waiting } pch = append(pch, &PieceChunk{ Status: s, NumPieces: psr.Length, }) totalPieces += psr.Length } ts.PieceChunks = pch } ts.Hash = t.InfoHash().String() ts.Name = t.Name() ts.TotalPieces = totalPieces return ts } const gap time.Duration = 2 * time.Second func (s *Torrent) returnPreviousMeasurements(now time.Time) bool { return now.Sub(s.gTime) < gap }