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.

mtproto_abridged.go 3.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. package wrappers
  2. import (
  3. "bytes"
  4. "io"
  5. "net"
  6. "github.com/juju/errors"
  7. "go.uber.org/zap"
  8. "github.com/9seconds/mtg/mtproto"
  9. "github.com/9seconds/mtg/utils"
  10. )
  11. const (
  12. mtprotoAbridgedSmallPacketLength = 0x7f
  13. mtprotoAbridgedQuickAckLength = 0x80
  14. mtprotoAbridgedLargePacketLength = 16777216 // 256 ^ 3
  15. )
  16. // MTProtoAbridged presents abridged connection between client and
  17. // middle proxy.
  18. type MTProtoAbridged struct {
  19. conn StreamReadWriteCloser
  20. opts *mtproto.ConnectionOpts
  21. logger *zap.SugaredLogger
  22. readCounter uint32
  23. writeCounter uint32
  24. }
  25. func (m *MTProtoAbridged) Read() ([]byte, error) {
  26. defer func() {
  27. m.readCounter++
  28. }()
  29. m.logger.Debugw("Read packet",
  30. "simple_ack", m.opts.ReadHacks.SimpleAck,
  31. "quick_ack", m.opts.ReadHacks.QuickAck,
  32. "counter", m.readCounter,
  33. )
  34. buf := &bytes.Buffer{}
  35. buf.Grow(3)
  36. if _, err := io.CopyN(buf, m.conn, 1); err != nil {
  37. return nil, errors.Annotate(err, "Cannot read message length")
  38. }
  39. msgLength := uint32(buf.Bytes()[0])
  40. buf.Reset()
  41. m.logger.Debugw("Packet first byte",
  42. "byte", msgLength,
  43. "counter", m.readCounter,
  44. "simple_ack", m.opts.ReadHacks.SimpleAck,
  45. "quick_ack", m.opts.ReadHacks.QuickAck,
  46. )
  47. if msgLength >= mtprotoAbridgedQuickAckLength {
  48. m.opts.ReadHacks.QuickAck = true
  49. msgLength -= mtprotoAbridgedQuickAckLength
  50. }
  51. if msgLength == mtprotoAbridgedSmallPacketLength {
  52. if _, err := io.CopyN(buf, m.conn, 3); err != nil {
  53. return nil, errors.Annotate(err, "Cannot read the correct message length")
  54. }
  55. number := utils.Uint24{}
  56. copy(number[:], buf.Bytes())
  57. msgLength = utils.FromUint24(number)
  58. }
  59. msgLength *= 4
  60. m.logger.Debugw("Packet length",
  61. "length", msgLength,
  62. "simple_ack", m.opts.ReadHacks.SimpleAck,
  63. "quick_ack", m.opts.ReadHacks.QuickAck,
  64. "counter", m.readCounter,
  65. )
  66. buf.Reset()
  67. buf.Grow(int(msgLength))
  68. if _, err := io.CopyN(buf, m.conn, int64(msgLength)); err != nil {
  69. return nil, errors.Annotate(err, "Cannot read message")
  70. }
  71. return buf.Bytes(), nil
  72. }
  73. func (m *MTProtoAbridged) Write(p []byte) (int, error) {
  74. defer func() {
  75. m.writeCounter++
  76. }()
  77. m.logger.Debugw("Write packet",
  78. "length", len(p),
  79. "simple_ack", m.opts.WriteHacks.SimpleAck,
  80. "quick_ack", m.opts.WriteHacks.QuickAck,
  81. "counter", m.writeCounter,
  82. )
  83. if len(p)%4 != 0 {
  84. return 0, errors.Errorf("Incorrect packet length %d", len(p))
  85. }
  86. if m.opts.WriteHacks.SimpleAck {
  87. return m.conn.Write(utils.ReverseBytes(p))
  88. }
  89. packetLength := len(p) / 4
  90. switch {
  91. case packetLength < mtprotoAbridgedSmallPacketLength:
  92. newData := append([]byte{byte(packetLength)}, p...)
  93. return m.conn.Write(newData)
  94. case packetLength < mtprotoAbridgedLargePacketLength:
  95. length24 := utils.ToUint24(uint32(packetLength))
  96. buf := &bytes.Buffer{}
  97. buf.Grow(1 + 3 + len(p))
  98. buf.WriteByte(byte(mtprotoAbridgedSmallPacketLength))
  99. buf.Write(length24[:])
  100. buf.Write(p)
  101. return m.conn.Write(buf.Bytes())
  102. }
  103. return 0, errors.Errorf("Packet is too big %d", len(p))
  104. }
  105. // Logger returns an instance of the logger for this wrapper.
  106. func (m *MTProtoAbridged) Logger() *zap.SugaredLogger {
  107. return m.logger
  108. }
  109. // LocalAddr returns local address of the underlying net.Conn.
  110. func (m *MTProtoAbridged) LocalAddr() *net.TCPAddr {
  111. return m.conn.LocalAddr()
  112. }
  113. // RemoteAddr returns remote address of the underlying net.Conn.
  114. func (m *MTProtoAbridged) RemoteAddr() *net.TCPAddr {
  115. return m.conn.RemoteAddr()
  116. }
  117. // Close closes underlying net.Conn instance.
  118. func (m *MTProtoAbridged) Close() error {
  119. return m.conn.Close()
  120. }
  121. // NewMTProtoAbridged creates new wrapper for abridged client connection.
  122. func NewMTProtoAbridged(conn StreamReadWriteCloser, opts *mtproto.ConnectionOpts) PacketReadWriteCloser {
  123. return &MTProtoAbridged{
  124. conn: conn,
  125. opts: opts,
  126. logger: conn.Logger().Named("mtproto-abridged"),
  127. }
  128. }