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.8KB

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