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
Du kan inte välja fler än 25 ämnen Ämnen måste starta med en bokstav eller siffra, kan innehålla bindestreck ('-') och vara max 35 tecken långa.

frame.go 3.1KB

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