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.

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