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
Nelze vybrat více než 25 témat Téma musí začínat písmenem nebo číslem, může obsahovat pomlčky („-“) a může být dlouhé až 35 znaků.

frame.go 3.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117
  1. package wrappers
  2. import (
  3. "bytes"
  4. "crypto/aes"
  5. "encoding/binary"
  6. "hash/crc32"
  7. "io"
  8. "net"
  9. "github.com/juju/errors"
  10. "github.com/9seconds/mtg/wrappers"
  11. )
  12. // Frame: { MessageLength(4) | SequenceNumber(4) | Message(???) | CRC32(4) [| padding(4), ...] }
  13. const (
  14. frameRWCMinMessageLength = 12
  15. frameRWCMaxMessageLength = 16777216
  16. )
  17. var frameRWCPadding = [4]byte{0x04, 0x00, 0x00, 0x00}
  18. type FrameRWC struct {
  19. wrappers.BufferedReader
  20. conn wrappers.ReadWriteCloserWithAddr
  21. readSeqNo int32
  22. writeSeqNo int32
  23. }
  24. func (f *FrameRWC) Write(buf []byte) (int, error) {
  25. writeBuf := &bytes.Buffer{}
  26. // 4 - len bytes
  27. // 4 - seq bytes
  28. // . - message
  29. // 4 - crc32
  30. messageLength := 4 + 4 + len(buf) + 4
  31. paddingLength := (aes.BlockSize - messageLength%aes.BlockSize) % aes.BlockSize
  32. writeBuf.Grow(messageLength + paddingLength)
  33. binary.Write(writeBuf, binary.LittleEndian, uint32(messageLength))
  34. binary.Write(writeBuf, binary.LittleEndian, f.writeSeqNo)
  35. writeBuf.Write(buf)
  36. f.writeSeqNo++
  37. checksum := crc32.ChecksumIEEE(writeBuf.Bytes())
  38. binary.Write(writeBuf, binary.LittleEndian, checksum)
  39. writeBuf.Write(bytes.Repeat(frameRWCPadding[:], paddingLength/4))
  40. _, err := f.conn.Write(writeBuf.Bytes())
  41. return len(buf), err
  42. }
  43. func (f *FrameRWC) Read(p []byte) (int, error) {
  44. return f.BufferedRead(p, func() error {
  45. buf := &bytes.Buffer{}
  46. for {
  47. buf.Reset()
  48. if _, err := io.CopyN(buf, f.conn, 4); err != nil {
  49. return 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 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 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 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 errors.Errorf("CRC32 checksum mismatch. Wait for %d, got %d", sum.Sum32(), checksum)
  77. }
  78. f.Buffer.Write(data)
  79. return nil
  80. })
  81. }
  82. func (f *FrameRWC) Close() error {
  83. return f.conn.Close()
  84. }
  85. func (f *FrameRWC) Addr() *net.TCPAddr {
  86. return f.conn.Addr()
  87. }
  88. func NewFrameRWC(conn wrappers.ReadWriteCloserWithAddr, seqNo int32) wrappers.ReadWriteCloserWithAddr {
  89. return &FrameRWC{
  90. BufferedReader: wrappers.NewBufferedReader(),
  91. conn: conn,
  92. readSeqNo: seqNo,
  93. writeSeqNo: seqNo,
  94. }
  95. }