1
0

udp.go 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. package proxy
  2. import (
  3. "io"
  4. "net"
  5. "strings"
  6. "sync"
  7. "time"
  8. "ehang.io/nps/bridge"
  9. "ehang.io/nps/lib/common"
  10. "ehang.io/nps/lib/conn"
  11. "ehang.io/nps/lib/file"
  12. "github.com/astaxie/beego/logs"
  13. )
  14. type UdpModeServer struct {
  15. BaseServer
  16. addrMap sync.Map
  17. listener *net.UDPConn
  18. }
  19. func NewUdpModeServer(bridge *bridge.Bridge, task *file.Tunnel) *UdpModeServer {
  20. s := new(UdpModeServer)
  21. s.bridge = bridge
  22. s.task = task
  23. return s
  24. }
  25. //开始
  26. func (s *UdpModeServer) Start() error {
  27. var err error
  28. if s.task.ServerIp == "" {
  29. s.task.ServerIp = "0.0.0.0"
  30. }
  31. s.listener, err = net.ListenUDP("udp", &net.UDPAddr{net.ParseIP(s.task.ServerIp), s.task.Port, ""})
  32. if err != nil {
  33. return err
  34. }
  35. for {
  36. buf := common.BufPoolUdp.Get().([]byte)
  37. n, addr, err := s.listener.ReadFromUDP(buf)
  38. if err != nil {
  39. if strings.Contains(err.Error(), "use of closed network connection") {
  40. break
  41. }
  42. continue
  43. }
  44. logs.Trace("New udp connection,client %d,remote address %s", s.task.Client.Id, addr)
  45. go s.process(addr, buf[:n])
  46. }
  47. return nil
  48. }
  49. func (s *UdpModeServer) process(addr *net.UDPAddr, data []byte) {
  50. if v, ok := s.addrMap.Load(addr.String()); ok {
  51. clientConn, ok := v.(io.ReadWriteCloser)
  52. if ok {
  53. clientConn.Write(data)
  54. s.task.Flow.Add(int64(len(data)), 0)
  55. }
  56. } else {
  57. if err := s.CheckFlowAndConnNum(s.task.Client); err != nil {
  58. logs.Warn("client id %d, task id %d,error %s, when udp connection", s.task.Client.Id, s.task.Id, err.Error())
  59. return
  60. }
  61. defer s.task.Client.AddConn()
  62. link := conn.NewLink(common.CONN_UDP, s.task.Target.TargetStr, s.task.Client.Cnf.Crypt, s.task.Client.Cnf.Compress, addr.String(), s.task.Target.LocalProxy)
  63. if clientConn, err := s.bridge.SendLinkInfo(s.task.Client.Id, link, s.task); err != nil {
  64. return
  65. } else {
  66. target := conn.GetConn(clientConn, s.task.Client.Cnf.Crypt, s.task.Client.Cnf.Compress, nil, true)
  67. s.addrMap.Store(addr.String(), target)
  68. defer target.Close()
  69. target.Write(data)
  70. buf := common.BufPoolUdp.Get().([]byte)
  71. defer common.BufPoolUdp.Put(buf)
  72. s.task.Flow.Add(int64(len(data)), 0)
  73. for {
  74. clientConn.SetReadDeadline(time.Now().Add(time.Minute * 10))
  75. if n, err := target.Read(buf); err != nil {
  76. s.addrMap.Delete(addr.String())
  77. logs.Warn(err)
  78. return
  79. } else {
  80. s.listener.WriteTo(buf[:n], addr)
  81. s.task.Flow.Add(0, int64(n))
  82. }
  83. }
  84. }
  85. }
  86. }
  87. func (s *UdpModeServer) Close() error {
  88. return s.listener.Close()
  89. }