proxy.go 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package tor
  2. import (
  3. std_bufio "bufio"
  4. "context"
  5. "crypto/rand"
  6. "encoding/hex"
  7. "net"
  8. "github.com/sagernet/sing-box/adapter"
  9. "github.com/sagernet/sing-box/log"
  10. "github.com/sagernet/sing/common"
  11. "github.com/sagernet/sing/common/auth"
  12. E "github.com/sagernet/sing/common/exceptions"
  13. M "github.com/sagernet/sing/common/metadata"
  14. N "github.com/sagernet/sing/common/network"
  15. "github.com/sagernet/sing/protocol/socks"
  16. "github.com/sagernet/sing/service"
  17. )
  18. type ProxyListener struct {
  19. ctx context.Context
  20. logger log.ContextLogger
  21. dialer N.Dialer
  22. connection adapter.ConnectionManager
  23. tcpListener *net.TCPListener
  24. username string
  25. password string
  26. authenticator *auth.Authenticator
  27. }
  28. func NewProxyListener(ctx context.Context, logger log.ContextLogger, dialer N.Dialer) *ProxyListener {
  29. var usernameB [64]byte
  30. var passwordB [64]byte
  31. rand.Read(usernameB[:])
  32. rand.Read(passwordB[:])
  33. username := hex.EncodeToString(usernameB[:])
  34. password := hex.EncodeToString(passwordB[:])
  35. return &ProxyListener{
  36. ctx: ctx,
  37. logger: logger,
  38. dialer: dialer,
  39. connection: service.FromContext[adapter.ConnectionManager](ctx),
  40. authenticator: auth.NewAuthenticator([]auth.User{{Username: username, Password: password}}),
  41. username: username,
  42. password: password,
  43. }
  44. }
  45. func (l *ProxyListener) Start() error {
  46. tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{
  47. IP: net.IPv4(127, 0, 0, 1),
  48. })
  49. if err != nil {
  50. return err
  51. }
  52. l.tcpListener = tcpListener
  53. go l.acceptLoop()
  54. return nil
  55. }
  56. func (l *ProxyListener) Port() uint16 {
  57. if l.tcpListener == nil {
  58. panic("start listener first")
  59. }
  60. return M.SocksaddrFromNet(l.tcpListener.Addr()).Port
  61. }
  62. func (l *ProxyListener) Username() string {
  63. return l.username
  64. }
  65. func (l *ProxyListener) Password() string {
  66. return l.password
  67. }
  68. func (l *ProxyListener) Close() error {
  69. return common.Close(l.tcpListener)
  70. }
  71. func (l *ProxyListener) acceptLoop() {
  72. for {
  73. tcpConn, err := l.tcpListener.AcceptTCP()
  74. if err != nil {
  75. return
  76. }
  77. ctx := log.ContextWithNewID(l.ctx)
  78. go func() {
  79. hErr := l.accept(ctx, tcpConn)
  80. if hErr != nil {
  81. if E.IsClosedOrCanceled(hErr) {
  82. l.logger.DebugContext(ctx, E.Cause(hErr, "proxy connection closed"))
  83. return
  84. }
  85. l.logger.ErrorContext(ctx, E.Cause(hErr, "proxy"))
  86. }
  87. }()
  88. }
  89. }
  90. func (l *ProxyListener) accept(ctx context.Context, conn *net.TCPConn) error {
  91. return socks.HandleConnectionEx(ctx, conn, std_bufio.NewReader(conn), l.authenticator, l, nil, M.SocksaddrFromNet(conn.RemoteAddr()), nil)
  92. }
  93. func (l *ProxyListener) NewConnectionEx(ctx context.Context, conn net.Conn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
  94. var metadata adapter.InboundContext
  95. metadata.Source = source
  96. metadata.Destination = destination
  97. metadata.Network = N.NetworkTCP
  98. l.logger.InfoContext(ctx, "proxy connection to ", metadata.Destination)
  99. l.connection.NewConnection(ctx, l.dialer, conn, metadata, onClose)
  100. }
  101. func (l *ProxyListener) NewPacketConnectionEx(ctx context.Context, conn N.PacketConn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
  102. var metadata adapter.InboundContext
  103. metadata.Source = source
  104. metadata.Destination = destination
  105. metadata.Network = N.NetworkUDP
  106. l.logger.InfoContext(ctx, "proxy packet connection to ", metadata.Destination)
  107. l.connection.NewPacketConnection(ctx, l.dialer, conn, metadata, onClose)
  108. }