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.

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