]> Sergey Matveev's repositories - btrtrc.git/blob - tracker/tracker.go
Add ability to set DialContext/ListenPacket for tracker announcements (#760)
[btrtrc.git] / tracker / tracker.go
1 package tracker
2
3 import (
4         "context"
5         "errors"
6         "fmt"
7         "net"
8         "net/http"
9         "net/url"
10         "time"
11
12         "github.com/anacrolix/dht/v2/krpc"
13         "github.com/anacrolix/log"
14         trHttp "github.com/anacrolix/torrent/tracker/http"
15         "github.com/anacrolix/torrent/tracker/shared"
16         "github.com/anacrolix/torrent/tracker/udp"
17 )
18
19 const (
20         None      = shared.None
21         Started   = shared.Started
22         Stopped   = shared.Stopped
23         Completed = shared.Completed
24 )
25
26 type AnnounceRequest = udp.AnnounceRequest
27
28 type AnnounceResponse = trHttp.AnnounceResponse
29
30 type Peer = trHttp.Peer
31
32 type AnnounceEvent = udp.AnnounceEvent
33
34 var ErrBadScheme = errors.New("unknown scheme")
35
36 type Announce struct {
37         TrackerUrl   string
38         Request      AnnounceRequest
39         HostHeader   string
40         HTTPProxy    func(*http.Request) (*url.URL, error)
41         DialContext  func(ctx context.Context, network, addr string) (net.Conn, error)
42         ListenPacket func(network, addr string) (net.PacketConn, error)
43         ServerName   string
44         UserAgent    string
45         UdpNetwork   string
46         // If the port is zero, it's assumed to be the same as the Request.Port.
47         ClientIp4 krpc.NodeAddr
48         // If the port is zero, it's assumed to be the same as the Request.Port.
49         ClientIp6 krpc.NodeAddr
50         Context   context.Context
51         Logger    log.Logger
52 }
53
54 // The code *is* the documentation.
55 const DefaultTrackerAnnounceTimeout = 15 * time.Second
56
57 func (me Announce) Do() (res AnnounceResponse, err error) {
58         cl, err := NewClient(me.TrackerUrl, NewClientOpts{
59                 Http: trHttp.NewClientOpts{
60                         Proxy:       me.HTTPProxy,
61                         DialContext: me.DialContext,
62                         ServerName:  me.ServerName,
63                 },
64                 UdpNetwork:   me.UdpNetwork,
65                 Logger:       me.Logger.WithContextValue(fmt.Sprintf("tracker client for %q", me.TrackerUrl)),
66                 ListenPacket: me.ListenPacket,
67         })
68         if err != nil {
69                 return
70         }
71         defer cl.Close()
72         if me.Context == nil {
73                 // This is just to maintain the old behaviour that should be a timeout of 15s. Users can
74                 // override it by providing their own Context. See comments elsewhere about longer timeouts
75                 // acting as rate limiting overloaded trackers.
76                 ctx, cancel := context.WithTimeout(context.Background(), DefaultTrackerAnnounceTimeout)
77                 defer cancel()
78                 me.Context = ctx
79         }
80         return cl.Announce(me.Context, me.Request, trHttp.AnnounceOpt{
81                 UserAgent:  me.UserAgent,
82                 HostHeader: me.HostHeader,
83                 ClientIp4:  me.ClientIp4.IP,
84                 ClientIp6:  me.ClientIp6.IP,
85         })
86 }