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
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

mtproto_proxy.go 3.6KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  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. type MTProtoProxy struct {
  12. conn PacketReadWriteCloser
  13. req *rpc.ProxyRequest
  14. logger *zap.SugaredLogger
  15. readCounter uint32
  16. writeCounter uint32
  17. }
  18. func (m *MTProtoProxy) Read() ([]byte, error) {
  19. defer func() {
  20. m.readCounter++
  21. }()
  22. m.logger.Debugw("Read packet",
  23. "counter", m.readCounter,
  24. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  25. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  26. )
  27. packet, err := m.conn.Read()
  28. if err != nil {
  29. return nil, errors.Annotate(err, "Cannot read packet")
  30. }
  31. m.logger.Debugw("Read packet length",
  32. "counter", m.readCounter,
  33. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  34. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  35. "length", len(packet),
  36. )
  37. if len(packet) < 4 {
  38. return nil, errors.Annotate(err, "Incorrect packet length")
  39. }
  40. tag, packet := packet[:4], packet[4:]
  41. switch {
  42. case bytes.Equal(tag, rpc.TagProxyAns):
  43. return m.readProxyAns(packet)
  44. case bytes.Equal(tag, rpc.TagSimpleAck):
  45. return m.readSimpleAck(packet)
  46. case bytes.Equal(tag, rpc.TagCloseExt):
  47. return m.readCloseExt(packet)
  48. }
  49. return nil, errors.Errorf("Unknown RPC answer %v", tag)
  50. }
  51. func (m *MTProtoProxy) readProxyAns(data []byte) ([]byte, error) {
  52. if len(data) < 12 {
  53. return nil, errors.Errorf("Incorrect data of proxy answer: %d", len(data))
  54. }
  55. data = data[12:]
  56. m.logger.Debugw("Read RPC_PROXY_ANS",
  57. "counter", m.readCounter,
  58. "length", len(data),
  59. )
  60. return data, nil
  61. }
  62. func (m *MTProtoProxy) readSimpleAck(data []byte) ([]byte, error) {
  63. if len(data) != 12 {
  64. return nil, errors.Errorf("Incorrect data of simple ack: %d", len(data))
  65. }
  66. data = data[8:12]
  67. m.logger.Debugw("Read RPC_SIMPLE_ACK",
  68. "counter", m.readCounter,
  69. "length", len(data),
  70. )
  71. return data, nil
  72. }
  73. func (m *MTProtoProxy) readCloseExt(data []byte) ([]byte, error) {
  74. m.logger.Debugw("Read RPC_CLOSE_EXT", "counter", m.readCounter)
  75. return nil, errors.New("Connection has been closed remotely by RPC call")
  76. }
  77. func (m *MTProtoProxy) Write(p []byte) (int, error) {
  78. defer func() {
  79. m.writeCounter++
  80. }()
  81. m.logger.Debugw("Write packet",
  82. "length", len(p),
  83. "counter", m.writeCounter,
  84. "simple_ack", m.req.Options.ReadHacks.SimpleAck,
  85. "quick_ack", m.req.Options.ReadHacks.QuickAck,
  86. )
  87. header := m.req.MakeHeader(p)
  88. if ce := m.logger.Desugar().Check(zap.DebugLevel, "RPC_PROXY_REQ header"); ce != nil {
  89. ce.Write(
  90. zap.Int("length", len(p)),
  91. zap.Uint32("counter", m.writeCounter),
  92. zap.Bool("simple_ack", m.req.Options.ReadHacks.QuickAck),
  93. zap.Bool("quick_ack", m.req.Options.ReadHacks.SimpleAck),
  94. zap.String("header", fmt.Sprintf("%v", header.Bytes())),
  95. )
  96. }
  97. header.Write(p)
  98. if _, err := m.conn.Write(header.Bytes()); err != nil {
  99. return 0, err
  100. }
  101. return len(p), nil
  102. }
  103. func (m *MTProtoProxy) Logger() *zap.SugaredLogger {
  104. return m.logger
  105. }
  106. func (m *MTProtoProxy) LocalAddr() *net.TCPAddr {
  107. return m.conn.LocalAddr()
  108. }
  109. func (m *MTProtoProxy) RemoteAddr() *net.TCPAddr {
  110. return m.conn.RemoteAddr()
  111. }
  112. func (m *MTProtoProxy) Close() error {
  113. return m.conn.Close()
  114. }
  115. func NewMTProtoProxy(conn PacketReadWriteCloser, connOpts *mtproto.ConnectionOpts, adTag []byte) (PacketReadWriteCloser, error) {
  116. req, err := rpc.NewProxyRequest(connOpts.ClientAddr, conn.LocalAddr(), connOpts, adTag)
  117. if err != nil {
  118. return nil, errors.Annotate(err, "Cannot create new RPC proxy request")
  119. }
  120. return &MTProtoProxy{
  121. conn: conn,
  122. logger: conn.Logger().Named("mtproto-proxy"),
  123. req: req,
  124. }, nil
  125. }