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.

proxy_request.go 2.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. package rpc
  2. import (
  3. "bytes"
  4. "crypto/rand"
  5. "encoding/binary"
  6. "fmt"
  7. "net"
  8. "github.com/juju/errors"
  9. "github.com/9seconds/mtg/mtproto"
  10. )
  11. // ProxyRequest is the data type for storing data required to compose
  12. // RPC_PROXY_REQ request.
  13. type ProxyRequest struct {
  14. Flags proxyRequestFlags
  15. ConnectionID []byte
  16. OurIPPort []byte
  17. ClientIPPort []byte
  18. ADTag []byte
  19. Options *mtproto.ConnectionOpts
  20. }
  21. // MakeHeader makes RPC_PROXY_REQ header. We need only to append the
  22. // data for it.
  23. func (r *ProxyRequest) MakeHeader(message []byte) (*bytes.Buffer, fmt.Stringer) {
  24. bufferLength := len(TagProxyRequest) +
  25. 4 + // len(flags)
  26. len(r.ConnectionID) +
  27. len(r.ClientIPPort) +
  28. len(r.OurIPPort) +
  29. len(ProxyRequestExtraSize) +
  30. len(ProxyRequestProxyTag) +
  31. 1 + // len(AdTag)
  32. len(r.ADTag)
  33. bufferLength += bufferLength % 4
  34. buf := &bytes.Buffer{}
  35. buf.Grow(bufferLength + len(message))
  36. flags := r.Flags
  37. if r.Options.ReadHacks.QuickAck {
  38. flags |= proxyRequestFlagsQuickAck
  39. }
  40. if bytes.HasPrefix(message, proxyRequestFlagsEncryptedPrefix[:]) {
  41. flags |= proxyRequestFlagsEncrypted
  42. }
  43. buf.Write(TagProxyRequest)
  44. buf.Write(flags.Bytes())
  45. buf.Write(r.ConnectionID)
  46. buf.Write(r.ClientIPPort)
  47. buf.Write(r.OurIPPort)
  48. buf.Write(ProxyRequestExtraSize)
  49. buf.Write(ProxyRequestProxyTag)
  50. buf.WriteByte(byte(len(r.ADTag)))
  51. buf.Write(r.ADTag)
  52. buf.Write(make([]byte, (4-buf.Len()%4)%4))
  53. return buf, flags
  54. }
  55. // NewProxyRequest build new ProxyRequest data structure.
  56. func NewProxyRequest(clientAddr, ownAddr *net.TCPAddr, opts *mtproto.ConnectionOpts, adTag []byte) (*ProxyRequest, error) {
  57. flags := proxyRequestFlagsHasAdTag | proxyRequestFlagsMagic | proxyRequestFlagsExtMode2
  58. if opts.ConnectionType == mtproto.ConnectionTypeAbridged {
  59. flags |= proxyRequestFlagsAbdridged
  60. } else {
  61. flags |= proxyRequestFlagsIntermediate
  62. }
  63. request := &ProxyRequest{
  64. Flags: flags,
  65. ADTag: adTag,
  66. Options: opts,
  67. ConnectionID: make([]byte, 8),
  68. ClientIPPort: make([]byte, 16+4),
  69. OurIPPort: make([]byte, 16+4),
  70. }
  71. if _, err := rand.Read(request.ConnectionID); err != nil {
  72. return nil, errors.Annotate(err, "Cannot generate connection ID")
  73. }
  74. port := [4]byte{}
  75. copy(request.ClientIPPort[:16], clientAddr.IP.To16())
  76. binary.LittleEndian.PutUint32(port[:], uint32(clientAddr.Port))
  77. copy(request.ClientIPPort[16:], port[:])
  78. copy(request.OurIPPort[:16], ownAddr.IP.To16())
  79. binary.LittleEndian.PutUint32(port[:], uint32(ownAddr.Port))
  80. copy(request.OurIPPort[16:], port[:])
  81. return request, nil
  82. }