proxy.go 3.2 KB

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