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
Vous ne pouvez pas sélectionner plus de 25 sujets Les noms de sujets doivent commencer par une lettre ou un nombre, peuvent contenir des tirets ('-') et peuvent comporter jusqu'à 35 caractères.

proxy.go 3.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. package proxy
  2. import (
  3. "io"
  4. "net"
  5. "sync"
  6. "github.com/juju/errors"
  7. uuid "github.com/satori/go.uuid"
  8. "go.uber.org/zap"
  9. "github.com/9seconds/mtg/client"
  10. "github.com/9seconds/mtg/config"
  11. "github.com/9seconds/mtg/mtproto"
  12. "github.com/9seconds/mtg/stats"
  13. "github.com/9seconds/mtg/telegram"
  14. "github.com/9seconds/mtg/wrappers"
  15. )
  16. type Proxy struct {
  17. clientInit client.Init
  18. tg telegram.Telegram
  19. conf *config.Config
  20. }
  21. func (p *Proxy) Serve() error {
  22. lsock, err := net.Listen("tcp", p.conf.BindAddr())
  23. if err != nil {
  24. return errors.Annotate(err, "Cannot create listen socket")
  25. }
  26. for {
  27. if conn, err := lsock.Accept(); err != nil {
  28. zap.S().Errorw("Cannot allocate incoming connection", "error", err)
  29. } else {
  30. go p.accept(conn)
  31. }
  32. }
  33. }
  34. func (p *Proxy) accept(conn net.Conn) {
  35. connID := uuid.NewV4().String()
  36. log := zap.S().With("connection_id", connID).Named("main")
  37. defer func() {
  38. conn.Close()
  39. if err := recover(); err != nil {
  40. stats.NewCrash()
  41. log.Errorw("Crash of accept handler", "error", err)
  42. }
  43. }()
  44. log.Infow("Client connected", "addr", conn.RemoteAddr())
  45. client, opts, err := p.clientInit(conn, connID, p.conf)
  46. if err != nil {
  47. log.Errorw("Cannot initialize client connection", "error", err)
  48. return
  49. }
  50. defer client.(io.Closer).Close()
  51. stats.ClientConnected(opts.ConnectionType, client.RemoteAddr())
  52. defer stats.ClientDisconnected(opts.ConnectionType, client.RemoteAddr())
  53. server, err := p.getTelegramConn(opts, connID)
  54. if err != nil {
  55. log.Errorw("Cannot initialize server connection", "error", err)
  56. return
  57. }
  58. defer server.(io.Closer).Close()
  59. wait := &sync.WaitGroup{}
  60. wait.Add(2)
  61. if p.conf.UseMiddleProxy() {
  62. clientPacket := client.(wrappers.PacketReadWriteCloser)
  63. serverPacket := server.(wrappers.PacketReadWriteCloser)
  64. go p.middlePipe(clientPacket, serverPacket, wait, &opts.ReadHacks)
  65. go p.middlePipe(serverPacket, clientPacket, wait, &opts.WriteHacks)
  66. } else {
  67. clientStream := client.(wrappers.StreamReadWriteCloser)
  68. serverStream := server.(wrappers.StreamReadWriteCloser)
  69. go p.directPipe(clientStream, serverStream, wait)
  70. go p.directPipe(serverStream, clientStream, wait)
  71. }
  72. wait.Wait()
  73. log.Infow("Client disconnected", "addr", conn.RemoteAddr())
  74. }
  75. func (p *Proxy) getTelegramConn(opts *mtproto.ConnectionOpts, connID string) (wrappers.Wrap, error) {
  76. streamConn, err := p.tg.Dial(connID, opts)
  77. if err != nil {
  78. return nil, errors.Annotate(err, "Cannot dial to Telegram")
  79. }
  80. packetConn, err := p.tg.Init(opts, streamConn)
  81. if err != nil {
  82. return nil, errors.Annotate(err, "Cannot handshake telegram")
  83. }
  84. return packetConn, nil
  85. }
  86. func (p *Proxy) middlePipe(src wrappers.PacketReadCloser, dst wrappers.PacketWriteCloser, wait *sync.WaitGroup, hacks *mtproto.Hacks) {
  87. defer func() {
  88. src.Close()
  89. dst.Close()
  90. wait.Done()
  91. }()
  92. for {
  93. hacks.SimpleAck = false
  94. hacks.QuickAck = false
  95. packet, err := src.Read()
  96. if err != nil {
  97. src.Logger().Warnw("Cannot read packet", "error", err)
  98. return
  99. }
  100. if _, err = dst.Write(packet); err != nil {
  101. src.Logger().Warnw("Cannot write packet", "error", err)
  102. return
  103. }
  104. }
  105. }
  106. func (p *Proxy) directPipe(src wrappers.StreamReadCloser, dst wrappers.StreamWriteCloser, wait *sync.WaitGroup) {
  107. defer func() {
  108. src.Close()
  109. dst.Close()
  110. wait.Done()
  111. }()
  112. if _, err := io.Copy(dst, src); err != nil {
  113. src.Logger().Warnw("Cannot pump sockets", "error", err)
  114. }
  115. }
  116. func NewProxy(conf *config.Config) *Proxy {
  117. var clientInit client.Init
  118. var tg telegram.Telegram
  119. if conf.UseMiddleProxy() {
  120. clientInit = client.MiddleInit
  121. tg = telegram.NewMiddleTelegram(conf)
  122. } else {
  123. clientInit = client.DirectInit
  124. tg = telegram.NewDirectTelegram(conf)
  125. }
  126. return &Proxy{
  127. conf: conf,
  128. clientInit: clientInit,
  129. tg: tg,
  130. }
  131. }