Highly-opinionated (ex-bullshit-free) MTPROTO proxy for Telegram. If you use v1.0 or upgrade broke you proxy, please read the chapter Version 2
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

proxy.go 3.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. package proxy
  2. import (
  3. "context"
  4. "io"
  5. "net"
  6. "sync"
  7. "go.uber.org/zap"
  8. "github.com/9seconds/mtg/config"
  9. "github.com/9seconds/mtg/protocol"
  10. "github.com/9seconds/mtg/stats"
  11. "github.com/9seconds/mtg/telegram"
  12. "github.com/9seconds/mtg/utils"
  13. "github.com/9seconds/mtg/wrappers"
  14. )
  15. const directPipeBufferSize = 1024 * 1024
  16. type Proxy struct {
  17. Logger *zap.SugaredLogger
  18. ClientProtocolMaker protocol.ClientProtocolMaker
  19. TelegramProtocolMaker protocol.TelegramProtocolMaker
  20. TelegramDialer telegram.Telegram
  21. }
  22. func (p *Proxy) Serve(listener net.Listener) {
  23. for {
  24. conn, err := listener.Accept()
  25. if err != nil {
  26. p.Logger.Errorw("Cannot allocate incoming connection", "error", err)
  27. continue
  28. }
  29. go p.accept(conn)
  30. }
  31. }
  32. func (p *Proxy) accept(conn net.Conn) {
  33. defer func() {
  34. conn.Close()
  35. if err := recover(); err != nil {
  36. stats.S.Crash()
  37. p.Logger.Errorw("Crash of accept handler", "error", err)
  38. }
  39. }()
  40. connID := wrappers.NewConnID()
  41. logger := p.Logger.With("connection_id", connID)
  42. if err := utils.InitTCP(conn); err != nil {
  43. logger.Errorw("Cannot initialize client TCP connection", "error", err)
  44. return
  45. }
  46. ctx, cancel := context.WithCancel(context.Background())
  47. defer cancel()
  48. wrappedConn := wrappers.NewClientConn(ctx, cancel, conn, connID)
  49. wrappedConn = wrappers.NewTraffic(wrappedConn)
  50. defer wrappedConn.Close()
  51. clientProtocol := p.ClientProtocolMaker()
  52. wrappedConn, err := clientProtocol.Handshake(wrappedConn)
  53. if err != nil {
  54. logger.Warnw("Cannot perform client handshake", "error", err)
  55. return
  56. }
  57. defer wrappedConn.Close()
  58. stats.S.ClientConnected(clientProtocol.GetConnectionType(), wrappedConn.RemoteAddr())
  59. defer stats.S.ClientDisconnected(clientProtocol.GetConnectionType(), wrappedConn.RemoteAddr())
  60. logger.Infow("Client connected", "addr", conn.RemoteAddr())
  61. req := &protocol.TelegramRequest{
  62. Logger: logger,
  63. ClientConn: wrappedConn,
  64. ConnID: connID,
  65. Ctx: ctx,
  66. Cancel: cancel,
  67. ClientProtocol: clientProtocol,
  68. }
  69. if len(config.C.AdTag) > 0 {
  70. err = p.acceptMiddleProxyConnection(req)
  71. } else {
  72. err = p.acceptDirectConnection(req)
  73. }
  74. logger.Infow("Client disconnected", "error", err, "addr", conn.RemoteAddr())
  75. }
  76. func (p *Proxy) acceptDirectConnection(request *protocol.TelegramRequest) error {
  77. telegramProtocol := p.TelegramProtocolMaker(p.TelegramDialer)
  78. telegramConnRaw, err := telegramProtocol.Handshake(request)
  79. if err != nil {
  80. return err
  81. }
  82. telegramConn := telegramConnRaw.(wrappers.StreamReadWriteCloser)
  83. defer telegramConn.Close()
  84. wg := &sync.WaitGroup{}
  85. wg.Add(2)
  86. go p.directPipe(telegramConn, request.ClientConn, wg, request.Logger)
  87. go p.directPipe(request.ClientConn, telegramConn, wg, request.Logger)
  88. <-request.Ctx.Done()
  89. wg.Wait()
  90. return request.Ctx.Err()
  91. }
  92. func (p *Proxy) directPipe(dst io.Writer,
  93. src io.Reader,
  94. wg *sync.WaitGroup,
  95. logger *zap.SugaredLogger) {
  96. defer wg.Done()
  97. buf := make([]byte, directPipeBufferSize)
  98. if _, err := io.CopyBuffer(dst, src, buf); err != nil {
  99. logger.Debugw("Cannot pump sockets", "error", err)
  100. }
  101. }
  102. func (p *Proxy) acceptMiddleProxyConnection(request *protocol.TelegramRequest) error {
  103. return nil
  104. }