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
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

mtproto_proxy.go 4.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. package wrappers
  2. import (
  3. "bytes"
  4. "fmt"
  5. "net"
  6. "github.com/juju/errors"
  7. "go.uber.org/zap"
  8. "github.com/9seconds/mtg/mtproto"
  9. "github.com/9seconds/mtg/mtproto/rpc"
  10. )
  11. // MTProtoProxy is a wrapper which creates/reads RPC responses from Telegram.
  12. type MTProtoProxy struct {
  13. conn PacketReadWriteCloser
  14. req *rpc.ProxyRequest
  15. logger *zap.SugaredLogger
  16. readCounter uint32
  17. writeCounter uint32
  18. }
  19. func (m *MTProtoProxy) Read() ([]byte, error) {
  20. defer func() {
  21. m.readCounter++
  22. }()
  23. m.logger.Debugw("Read packet",
  24. "counter", m.readCounter,
  25. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  26. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  27. )
  28. packet, err := m.conn.Read()
  29. if err != nil {
  30. return nil, errors.Annotate(err, "Cannot read packet")
  31. }
  32. m.logger.Debugw("Read packet length",
  33. "counter", m.readCounter,
  34. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  35. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  36. "length", len(packet),
  37. )
  38. if len(packet) < 4 {
  39. return nil, errors.Annotate(err, "Incorrect packet length")
  40. }
  41. tag, packet := packet[:4], packet[4:]
  42. switch {
  43. case bytes.Equal(tag, rpc.TagProxyAns):
  44. return m.readProxyAns(packet)
  45. case bytes.Equal(tag, rpc.TagSimpleAck):
  46. return m.readSimpleAck(packet)
  47. case bytes.Equal(tag, rpc.TagCloseExt):
  48. return m.readCloseExt()
  49. }
  50. return nil, errors.Errorf("Unknown RPC answer %v", tag)
  51. }
  52. func (m *MTProtoProxy) readProxyAns(data []byte) ([]byte, error) {
  53. if len(data) < 12 {
  54. return nil, errors.Errorf("Incorrect data of proxy answer: %d", len(data))
  55. }
  56. data = data[12:]
  57. m.logger.Debugw("Read RPC_PROXY_ANS",
  58. "counter", m.readCounter,
  59. "length", len(data),
  60. )
  61. return data, nil
  62. }
  63. func (m *MTProtoProxy) readSimpleAck(data []byte) ([]byte, error) {
  64. if len(data) != 12 {
  65. return nil, errors.Errorf("Incorrect data of simple ack: %d", len(data))
  66. }
  67. data = data[8:12]
  68. m.req.Options.WriteHacks.SimpleAck = true
  69. m.logger.Debugw("Read RPC_SIMPLE_ACK",
  70. "counter", m.readCounter,
  71. "length", len(data),
  72. )
  73. return data, nil
  74. }
  75. func (m *MTProtoProxy) readCloseExt() ([]byte, error) {
  76. m.logger.Debugw("Read RPC_CLOSE_EXT", "counter", m.readCounter)
  77. return nil, errors.New("Connection has been closed remotely by RPC call")
  78. }
  79. func (m *MTProtoProxy) Write(p []byte) (int, error) {
  80. defer func() {
  81. m.writeCounter++
  82. }()
  83. m.logger.Debugw("Write packet",
  84. "length", len(p),
  85. "counter", m.writeCounter,
  86. "simple_ack", m.req.Options.ReadHacks.SimpleAck,
  87. "quick_ack", m.req.Options.ReadHacks.QuickAck,
  88. )
  89. header, flags := m.req.MakeHeader(p)
  90. if ce := m.logger.Desugar().Check(zap.DebugLevel, "RPC_PROXY_REQ header"); ce != nil {
  91. ce.Write(
  92. zap.Int("length", len(p)),
  93. zap.Uint32("counter", m.writeCounter),
  94. zap.Bool("simple_ack", m.req.Options.ReadHacks.QuickAck),
  95. zap.Bool("quick_ack", m.req.Options.ReadHacks.SimpleAck),
  96. zap.String("header", fmt.Sprintf("%v", header.Bytes())),
  97. zap.Stringer("flags", flags),
  98. )
  99. }
  100. header.Write(p) // nolint: gosec
  101. if _, err := m.conn.Write(header.Bytes()); err != nil {
  102. return 0, err
  103. }
  104. return len(p), nil
  105. }
  106. // Logger returns an instance of the logger for this wrapper.
  107. func (m *MTProtoProxy) Logger() *zap.SugaredLogger {
  108. return m.logger
  109. }
  110. // LocalAddr returns local address of the underlying net.Conn.
  111. func (m *MTProtoProxy) LocalAddr() *net.TCPAddr {
  112. return m.conn.LocalAddr()
  113. }
  114. // RemoteAddr returns remote address of the underlying net.Conn.
  115. func (m *MTProtoProxy) RemoteAddr() *net.TCPAddr {
  116. return m.conn.RemoteAddr()
  117. }
  118. // Close closes underlying net.Conn instance.
  119. func (m *MTProtoProxy) Close() error {
  120. return m.conn.Close()
  121. }
  122. // NewMTProtoProxy creates new RPC wrapper.
  123. func NewMTProtoProxy(conn PacketReadWriteCloser, connOpts *mtproto.ConnectionOpts,
  124. adTag []byte) (PacketReadWriteCloser, error) {
  125. req, err := rpc.NewProxyRequest(connOpts.ClientAddr, conn.LocalAddr(), connOpts, adTag)
  126. if err != nil {
  127. return nil, errors.Annotate(err, "Cannot create new RPC proxy request")
  128. }
  129. return &MTProtoProxy{
  130. conn: conn,
  131. logger: conn.Logger().Named("mtproto-proxy"),
  132. req: req,
  133. }, nil
  134. }