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.

frame.go 3.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131
  1. package wrappers
  2. import (
  3. "bytes"
  4. "crypto/aes"
  5. "encoding/binary"
  6. "fmt"
  7. "hash/crc32"
  8. "io"
  9. "io/ioutil"
  10. "net"
  11. "github.com/juju/errors"
  12. "github.com/9seconds/mtg/wrappers"
  13. )
  14. // Frame: { MessageLength(4) | SequenceNumber(4) | Message(???) | CRC32(4) [| padding(4), ...] }
  15. const (
  16. frameRWCMinMessageLength = 12
  17. frameRWCMaxMessageLength = 16777216
  18. )
  19. var frameRWCPadding = []byte{0x04, 0x00, 0x00, 0x00}
  20. type FrameRWC struct {
  21. wrappers.BufferedReader
  22. conn wrappers.ReadWriteCloserWithAddr
  23. readSeqNo int32
  24. writeSeqNo int32
  25. }
  26. func (f *FrameRWC) Write(buf []byte) (int, error) {
  27. writeBuf := &bytes.Buffer{}
  28. // 4 - len bytes
  29. // 4 - seq bytes
  30. // . - message
  31. // 4 - crc32
  32. messageLength := 4 + 4 + len(buf) + 4
  33. paddingLength := (aes.BlockSize - messageLength%aes.BlockSize) % aes.BlockSize
  34. writeBuf.Grow(messageLength + paddingLength)
  35. binary.Write(writeBuf, binary.LittleEndian, uint32(messageLength))
  36. binary.Write(writeBuf, binary.LittleEndian, f.writeSeqNo)
  37. writeBuf.Write(buf)
  38. f.writeSeqNo++
  39. checksum := crc32.ChecksumIEEE(writeBuf.Bytes())
  40. binary.Write(writeBuf, binary.LittleEndian, checksum)
  41. writeBuf.Write(bytes.Repeat(frameRWCPadding, paddingLength/4))
  42. _, err := f.conn.Write(writeBuf.Bytes())
  43. return len(buf), err
  44. }
  45. func (f *FrameRWC) Read(p []byte) (int, error) {
  46. return f.BufferedRead(p, func() error {
  47. buf := &bytes.Buffer{}
  48. sum := crc32.NewIEEE()
  49. writer := io.MultiWriter(buf, sum)
  50. for {
  51. buf.Reset()
  52. sum.Reset()
  53. if _, err := io.CopyN(writer, f.conn, 4); err != nil {
  54. return errors.Annotate(err, "Cannot read frame padding")
  55. }
  56. if !bytes.Equal(buf.Bytes(), frameRWCPadding[:]) {
  57. break
  58. }
  59. }
  60. messageLength := binary.LittleEndian.Uint32(buf.Bytes())
  61. if messageLength%4 != 0 || messageLength < frameRWCMinMessageLength || messageLength > frameRWCMaxMessageLength {
  62. return errors.Errorf("Incorrect frame message length %d", messageLength)
  63. }
  64. buf.Reset()
  65. buf.Grow(int(messageLength) - 4 - 4)
  66. if _, err := io.CopyN(writer, f.conn, int64(messageLength)-4-4); err != nil {
  67. return errors.Annotate(err, "Cannot read the message frame")
  68. }
  69. var seqNo int32
  70. binary.Read(buf, binary.LittleEndian, &seqNo)
  71. if seqNo != f.readSeqNo {
  72. return errors.Errorf("Unexpected sequence number %d (wait for %d)", seqNo, f.readSeqNo)
  73. }
  74. f.readSeqNo++
  75. data, _ := ioutil.ReadAll(buf)
  76. buf.Reset()
  77. // write to buf, not to writer. This is because we are going to fetch
  78. // crc32 checksum.
  79. if _, err := io.CopyN(buf, f.conn, 4); err != nil {
  80. return errors.Annotate(err, "Cannot read checksum")
  81. }
  82. checksum := binary.LittleEndian.Uint32(buf.Bytes())
  83. if checksum != sum.Sum32() {
  84. return errors.Errorf("CRC32 checksum mismatch. Wait for %d, got %d", sum.Sum32(), checksum)
  85. }
  86. f.Buffer.Write(data)
  87. return nil
  88. })
  89. }
  90. func (f *FrameRWC) Close() error {
  91. return f.conn.Close()
  92. }
  93. func (f *FrameRWC) LocalAddr() *net.TCPAddr {
  94. return f.conn.LocalAddr()
  95. }
  96. func (f *FrameRWC) RemoteAddr() *net.TCPAddr {
  97. return f.conn.RemoteAddr()
  98. }
  99. func NewFrameRWC(conn wrappers.ReadWriteCloserWithAddr, seqNo int32) wrappers.ReadWriteCloserWithAddr {
  100. return &FrameRWC{
  101. BufferedReader: wrappers.NewBufferedReader(),
  102. conn: conn,
  103. readSeqNo: seqNo,
  104. writeSeqNo: seqNo,
  105. }
  106. }