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
Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

proxy_request.go 2.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  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. switch opts.ConnectionType {
  59. case mtproto.ConnectionTypeAbridged:
  60. flags |= proxyRequestFlagsAbdridged
  61. case mtproto.ConnectionTypeIntermediate:
  62. flags |= proxyRequestFlagsIntermediate
  63. }
  64. request := &ProxyRequest{
  65. Flags: flags,
  66. ADTag: adTag,
  67. Options: opts,
  68. ConnectionID: make([]byte, 8),
  69. ClientIPPort: make([]byte, 16+4),
  70. OurIPPort: make([]byte, 16+4),
  71. }
  72. if _, err := rand.Read(request.ConnectionID); err != nil {
  73. return nil, errors.Annotate(err, "Cannot generate connection ID")
  74. }
  75. port := [4]byte{}
  76. copy(request.ClientIPPort[:16], clientAddr.IP.To16())
  77. binary.LittleEndian.PutUint32(port[:], uint32(clientAddr.Port))
  78. copy(request.ClientIPPort[16:], port[:])
  79. copy(request.OurIPPort[:16], ownAddr.IP.To16())
  80. binary.LittleEndian.PutUint32(port[:], uint32(ownAddr.Port))
  81. copy(request.OurIPPort[16:], port[:])
  82. return request, nil
  83. }