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 символов.

mtproto_proxy.go 3.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137
  1. package wrappers
  2. import (
  3. "bytes"
  4. "net"
  5. "github.com/juju/errors"
  6. "go.uber.org/zap"
  7. "github.com/9seconds/mtg/mtproto"
  8. "github.com/9seconds/mtg/mtproto/rpc"
  9. )
  10. type MTProtoProxy struct {
  11. conn PacketReadWriteCloser
  12. req *rpc.ProxyRequest
  13. logger *zap.SugaredLogger
  14. readCounter uint32
  15. writeCounter uint32
  16. }
  17. func (m *MTProtoProxy) Read() ([]byte, error) {
  18. m.logger.Debugw("Read packet",
  19. "counter", m.readCounter,
  20. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  21. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  22. )
  23. packet, err := m.conn.Read()
  24. if err != nil {
  25. return nil, errors.Annotate(err, "Cannot read packet")
  26. }
  27. m.logger.Debugw("Read packet length",
  28. "counter", m.readCounter,
  29. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  30. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  31. "length", len(packet),
  32. )
  33. if len(packet) < 4 {
  34. return nil, errors.Annotate(err, "Incorrect packet length")
  35. }
  36. tag, packet := packet[:4], packet[4:]
  37. m.logger.Debugw("Read RPC tag",
  38. "counter", m.readCounter,
  39. "simple_ack", m.req.Options.WriteHacks.SimpleAck,
  40. "quick_ack", m.req.Options.WriteHacks.QuickAck,
  41. "tag", tag,
  42. )
  43. m.readCounter++
  44. switch {
  45. case bytes.Equal(tag, rpc.TagProxyAns):
  46. return m.readProxyAns(packet)
  47. case bytes.Equal(tag, rpc.TagSimpleAck):
  48. return m.readSimpleAck(packet)
  49. case bytes.Equal(tag, rpc.TagCloseExt):
  50. return m.readCloseExt(packet)
  51. }
  52. return nil, errors.Errorf("Unknown RPC answer %v", tag)
  53. }
  54. func (m *MTProtoProxy) readProxyAns(data []byte) ([]byte, error) {
  55. if len(data) < 12 {
  56. return nil, errors.Errorf("Incorrect data of proxy answer: %d", len(data))
  57. }
  58. data = data[12:]
  59. m.logger.Debugw("Read RPC_PROXY_ANS", "length", len(data))
  60. return data, nil
  61. }
  62. func (m *MTProtoProxy) readSimpleAck(data []byte) ([]byte, error) {
  63. if len(data) != 12 {
  64. return nil, errors.Errorf("Incorrect data of simple ack: %d", len(data))
  65. }
  66. data = data[8:12]
  67. m.logger.Debugw("Read RPC_SIMPLE_ACK", "length", len(data))
  68. return data, nil
  69. }
  70. func (m *MTProtoProxy) readCloseExt(data []byte) ([]byte, error) {
  71. m.logger.Debugw("Read RPC_CLOSE_EXT")
  72. return nil, errors.New("Connection has been closed remotely by RPC call")
  73. }
  74. func (m *MTProtoProxy) Write(p []byte) (int, error) {
  75. m.logger.Debugw("Write packet",
  76. "length", len(p),
  77. "counter", m.writeCounter,
  78. "simple_ack", m.req.Options.ReadHacks.SimpleAck,
  79. "quick_ack", m.req.Options.ReadHacks.QuickAck,
  80. )
  81. m.writeCounter++
  82. if _, err := m.conn.Write(p); err != nil {
  83. return 0, err
  84. }
  85. return len(p), nil
  86. }
  87. func (m *MTProtoProxy) Logger() *zap.SugaredLogger {
  88. return m.logger
  89. }
  90. func (m *MTProtoProxy) LocalAddr() *net.TCPAddr {
  91. return m.conn.LocalAddr()
  92. }
  93. func (m *MTProtoProxy) RemoteAddr() *net.TCPAddr {
  94. return m.conn.RemoteAddr()
  95. }
  96. func (m *MTProtoProxy) Close() error {
  97. return m.conn.Close()
  98. }
  99. func NewMTProtoProxy(conn PacketReadWriteCloser, connOpts *mtproto.ConnectionOpts, adTag []byte) (PacketReadWriteCloser, error) {
  100. req, err := rpc.NewProxyRequest(connOpts.ClientAddr, conn.LocalAddr(), connOpts, adTag)
  101. if err != nil {
  102. return nil, errors.Annotate(err, "Cannot create new RPC proxy request")
  103. }
  104. return &MTProtoProxy{
  105. conn: conn,
  106. logger: conn.Logger().Named("mtproto-proxy"),
  107. req: req,
  108. }, nil
  109. }