conn-client.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. package udp
  2. import (
  3. "context"
  4. "net"
  5. "github.com/anacrolix/log"
  6. "github.com/anacrolix/missinggo/v2"
  7. )
  8. type listenPacketFunc func(network, addr string) (net.PacketConn, error)
  9. type NewConnClientOpts struct {
  10. // The network to operate to use, such as "udp4", "udp", "udp6".
  11. Network string
  12. // Tracker address
  13. Host string
  14. // If non-nil, forces either IPv4 or IPv6 in the UDP tracker wire protocol.
  15. Ipv6 *bool
  16. // Logger to use for internal errors.
  17. Logger log.Logger
  18. // Custom function to use as a substitute for net.ListenPacket
  19. ListenPacket listenPacketFunc
  20. }
  21. // Manages a Client with a specific connection.
  22. type ConnClient struct {
  23. Client Client
  24. conn net.PacketConn
  25. d Dispatcher
  26. readErr error
  27. closed bool
  28. newOpts NewConnClientOpts
  29. }
  30. func (cc *ConnClient) reader() {
  31. b := make([]byte, 0x800)
  32. for {
  33. n, addr, err := cc.conn.ReadFrom(b)
  34. if err != nil {
  35. // TODO: Do bad things to the dispatcher, and incoming calls to the client if we have a
  36. // read error.
  37. cc.readErr = err
  38. if !cc.closed {
  39. // don't panic, just close the connection, fix https://github.com/anacrolix/torrent/issues/845
  40. cc.Close()
  41. }
  42. break
  43. }
  44. err = cc.d.Dispatch(b[:n], addr)
  45. if err != nil {
  46. cc.newOpts.Logger.Levelf(log.Debug, "dispatching packet received on %v: %v", cc.conn.LocalAddr(), err)
  47. }
  48. }
  49. }
  50. func ipv6(opt *bool, network string, remoteAddr net.Addr) bool {
  51. if opt != nil {
  52. return *opt
  53. }
  54. switch network {
  55. case "udp4":
  56. return false
  57. case "udp6":
  58. return true
  59. }
  60. rip := missinggo.AddrIP(remoteAddr)
  61. return rip.To16() != nil && rip.To4() == nil
  62. }
  63. // Allows a UDP Client to write packets to an endpoint without knowing about the network specifics.
  64. type clientWriter struct {
  65. pc net.PacketConn
  66. network string
  67. address string
  68. }
  69. func (me clientWriter) Write(p []byte) (n int, err error) {
  70. addr, err := net.ResolveUDPAddr(me.network, me.address)
  71. if err != nil {
  72. return
  73. }
  74. return me.pc.WriteTo(p, addr)
  75. }
  76. func NewConnClient(opts NewConnClientOpts) (cc *ConnClient, err error) {
  77. var conn net.PacketConn
  78. if opts.ListenPacket != nil {
  79. conn, err = opts.ListenPacket(opts.Network, ":0")
  80. } else {
  81. conn, err = net.ListenPacket(opts.Network, ":0")
  82. }
  83. if err != nil {
  84. return
  85. }
  86. if opts.Logger.IsZero() {
  87. opts.Logger = log.Default
  88. }
  89. cc = &ConnClient{
  90. Client: Client{
  91. Writer: clientWriter{
  92. pc: conn,
  93. network: opts.Network,
  94. address: opts.Host,
  95. },
  96. },
  97. conn: conn,
  98. newOpts: opts,
  99. }
  100. cc.Client.Dispatcher = &cc.d
  101. go cc.reader()
  102. return
  103. }
  104. func (cc *ConnClient) Close() error {
  105. cc.closed = true
  106. return cc.conn.Close()
  107. }
  108. func (cc *ConnClient) Announce(
  109. ctx context.Context, req AnnounceRequest, opts Options,
  110. ) (
  111. h AnnounceResponseHeader, nas AnnounceResponsePeers, err error,
  112. ) {
  113. return cc.Client.Announce(ctx, req, opts, func(addr net.Addr) bool {
  114. return ipv6(cc.newOpts.Ipv6, cc.newOpts.Network, addr)
  115. })
  116. }
  117. func (cc *ConnClient) LocalAddr() net.Addr {
  118. return cc.conn.LocalAddr()
  119. }