bridge.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. package proxybridge
  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. "github.com/sagernet/sing/common/logger"
  14. M "github.com/sagernet/sing/common/metadata"
  15. N "github.com/sagernet/sing/common/network"
  16. "github.com/sagernet/sing/protocol/socks"
  17. "github.com/sagernet/sing/service"
  18. )
  19. type Bridge struct {
  20. ctx context.Context
  21. logger logger.ContextLogger
  22. tag string
  23. dialer N.Dialer
  24. connection adapter.ConnectionManager
  25. tcpListener *net.TCPListener
  26. username string
  27. password string
  28. authenticator *auth.Authenticator
  29. }
  30. func New(ctx context.Context, logger logger.ContextLogger, tag string, dialer N.Dialer) (*Bridge, error) {
  31. username := randomHex(16)
  32. password := randomHex(16)
  33. tcpListener, err := net.ListenTCP("tcp", &net.TCPAddr{IP: net.IPv4(127, 0, 0, 1)})
  34. if err != nil {
  35. return nil, err
  36. }
  37. bridge := &Bridge{
  38. ctx: ctx,
  39. logger: logger,
  40. tag: tag,
  41. dialer: dialer,
  42. connection: service.FromContext[adapter.ConnectionManager](ctx),
  43. tcpListener: tcpListener,
  44. username: username,
  45. password: password,
  46. authenticator: auth.NewAuthenticator([]auth.User{{Username: username, Password: password}}),
  47. }
  48. go bridge.acceptLoop()
  49. return bridge, nil
  50. }
  51. func randomHex(size int) string {
  52. raw := make([]byte, size)
  53. rand.Read(raw)
  54. return hex.EncodeToString(raw)
  55. }
  56. func (b *Bridge) Port() uint16 {
  57. return M.SocksaddrFromNet(b.tcpListener.Addr()).Port
  58. }
  59. func (b *Bridge) Username() string {
  60. return b.username
  61. }
  62. func (b *Bridge) Password() string {
  63. return b.password
  64. }
  65. func (b *Bridge) Close() error {
  66. return common.Close(b.tcpListener)
  67. }
  68. func (b *Bridge) acceptLoop() {
  69. for {
  70. tcpConn, err := b.tcpListener.AcceptTCP()
  71. if err != nil {
  72. return
  73. }
  74. ctx := log.ContextWithNewID(b.ctx)
  75. go func() {
  76. hErr := socks.HandleConnectionEx(ctx, tcpConn, std_bufio.NewReader(tcpConn), b.authenticator, b, nil, 0, M.SocksaddrFromNet(tcpConn.RemoteAddr()), nil)
  77. if hErr == nil {
  78. return
  79. }
  80. if E.IsClosedOrCanceled(hErr) {
  81. b.logger.DebugContext(ctx, E.Cause(hErr, b.tag, " connection closed"))
  82. return
  83. }
  84. b.logger.ErrorContext(ctx, E.Cause(hErr, b.tag))
  85. }()
  86. }
  87. }
  88. func (b *Bridge) NewConnectionEx(ctx context.Context, conn net.Conn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
  89. var metadata adapter.InboundContext
  90. metadata.Source = source
  91. metadata.Destination = destination
  92. metadata.Network = N.NetworkTCP
  93. b.logger.InfoContext(ctx, b.tag, " connection to ", metadata.Destination)
  94. b.connection.NewConnection(ctx, b.dialer, conn, metadata, onClose)
  95. }
  96. func (b *Bridge) NewPacketConnectionEx(ctx context.Context, conn N.PacketConn, source M.Socksaddr, destination M.Socksaddr, onClose N.CloseHandlerFunc) {
  97. var metadata adapter.InboundContext
  98. metadata.Source = source
  99. metadata.Destination = destination
  100. metadata.Network = N.NetworkUDP
  101. b.logger.InfoContext(ctx, b.tag, " packet connection to ", metadata.Destination)
  102. b.connection.NewPacketConnection(ctx, b.dialer, conn, metadata, onClose)
  103. }