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
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

proxy.go 3.2KB

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