2020-04-06 13:38:01 +08:00
|
|
|
package torrent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
2020-04-21 16:08:43 +08:00
|
|
|
"sync"
|
2020-04-06 14:45:47 +08:00
|
|
|
|
2020-04-21 16:08:43 +08:00
|
|
|
"github.com/anacrolix/log"
|
|
|
|
|
|
|
|
"github.com/anacrolix/torrent/tracker"
|
2020-04-06 14:45:47 +08:00
|
|
|
"github.com/anacrolix/torrent/webtorrent"
|
2020-04-21 16:08:43 +08:00
|
|
|
"github.com/pion/datachannel"
|
2020-04-06 13:38:01 +08:00
|
|
|
)
|
|
|
|
|
2020-04-22 09:42:31 +08:00
|
|
|
type websocketTrackerStatus struct {
|
2020-04-06 13:38:01 +08:00
|
|
|
url url.URL
|
2020-04-22 09:42:31 +08:00
|
|
|
tc *webtorrent.TrackerClient
|
2020-04-06 13:38:01 +08:00
|
|
|
}
|
|
|
|
|
2020-04-22 09:42:31 +08:00
|
|
|
func (me websocketTrackerStatus) statusLine() string {
|
|
|
|
return fmt.Sprintf("%q: %+v", me.tc.Url, me.tc.Stats())
|
2020-04-06 13:38:01 +08:00
|
|
|
}
|
|
|
|
|
2020-04-22 09:42:31 +08:00
|
|
|
func (me websocketTrackerStatus) URL() url.URL {
|
2020-04-06 13:38:01 +08:00
|
|
|
return me.url
|
|
|
|
}
|
2020-04-21 16:08:43 +08:00
|
|
|
|
|
|
|
type refCountedWebtorrentTrackerClient struct {
|
|
|
|
webtorrent.TrackerClient
|
|
|
|
refCount int
|
|
|
|
}
|
|
|
|
|
|
|
|
type websocketTrackers struct {
|
|
|
|
PeerId [20]byte
|
|
|
|
Logger log.Logger
|
|
|
|
GetAnnounceRequest func(event tracker.AnnounceEvent, infoHash [20]byte) tracker.AnnounceRequest
|
|
|
|
OnConn func(datachannel.ReadWriteCloser, webtorrent.DataChannelContext)
|
|
|
|
mu sync.Mutex
|
|
|
|
clients map[string]*refCountedWebtorrentTrackerClient
|
|
|
|
}
|
|
|
|
|
|
|
|
func (me *websocketTrackers) Get(url string) (*webtorrent.TrackerClient, func()) {
|
|
|
|
me.mu.Lock()
|
|
|
|
defer me.mu.Unlock()
|
|
|
|
value, ok := me.clients[url]
|
|
|
|
if !ok {
|
|
|
|
value = &refCountedWebtorrentTrackerClient{
|
|
|
|
TrackerClient: webtorrent.TrackerClient{
|
|
|
|
Url: url,
|
|
|
|
GetAnnounceRequest: me.GetAnnounceRequest,
|
|
|
|
PeerId: me.PeerId,
|
|
|
|
OnConn: me.OnConn,
|
|
|
|
Logger: me.Logger.WithText(func(m log.Msg) string {
|
|
|
|
return fmt.Sprintf("tracker client for %q: %v", url, m)
|
|
|
|
}),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
go func() {
|
|
|
|
err := value.TrackerClient.Run()
|
|
|
|
if err != nil {
|
|
|
|
me.Logger.Printf("error running tracker client for %q: %v", url, err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
if me.clients == nil {
|
|
|
|
me.clients = make(map[string]*refCountedWebtorrentTrackerClient)
|
|
|
|
}
|
|
|
|
me.clients[url] = value
|
|
|
|
}
|
|
|
|
value.refCount++
|
|
|
|
return &value.TrackerClient, func() {
|
|
|
|
me.mu.Lock()
|
|
|
|
defer me.mu.Unlock()
|
|
|
|
value.refCount--
|
|
|
|
if value.refCount == 0 {
|
|
|
|
value.TrackerClient.Close()
|
|
|
|
delete(me.clients, url)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|